Files
lockdown-iOS-mirror/LockdowniOS/HomeViewController.swift
2024-12-11 11:25:28 +02:00

1396 lines
66 KiB
Swift

//
// HomeViewController.swift
// Lockdown
//
// Created by Johnny Lin on 7/31/19.
// Copyright © 2019 Confirmed Inc. All rights reserved.
//
import Foundation
import NetworkExtension
import CocoaLumberjackSwift
import UIKit
import PromiseKit
import StoreKit
import PopupDialog
import AwesomeSpotlightView
import SwiftUI
class CircularView: UIView {
override func layoutSubviews() {
super.layoutSubviews()
self.layer.cornerRadius = self.bounds.size.width * 0.50
}
@IBInspectable var shadowUIColor: UIColor? {
didSet {
redrawShadowColor()
}
}
func redrawShadowColor() {
self.layer.shadowColor = shadowUIColor?.cgColor
}
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
super.traitCollectionDidChange(previousTraitCollection)
redrawShadowColor()
}
}
let kHasSeenEmailSignup = "hasSeenEmailSignup"
class HomeViewController: BaseViewController, AwesomeSpotlightViewDelegate, Loadable {
let kHasViewedTutorial = "hasViewedTutorial"
let kHasSeenInitialFirewallConnectedDialog = "hasSeenInitialFirewallConnectedDialog11"
let kHasSeenShare = "hasSeenShareDialog4"
let ratingCountKey = "ratingCount" + lastVersionToAskForRating
let ratingTriggeredKey = "ratingTriggered" + lastVersionToAskForRating
var feedbackFlow: FeedbackFlow?
private lazy var scrollView: UIScrollView = {
let view = UIScrollView()
view.isScrollEnabled = true
return view
}()
private lazy var contentView: UIView = {
let view = UIView()
return view
}()
private lazy var yourCurrentPlanLabel: UILabel = {
let label = UILabel()
label.text = NSLocalizedString("Your current plan is", comment: "")
label.font = fontRegular14
label.numberOfLines = 0
label.textColor = .label
return label
}()
private lazy var upgradeLabel: UIButton = {
let button = UIButton(type: .system)
button.tintColor = .white
button.setTitle(NSLocalizedString("See plans", comment: ""), for: .normal)
button.titleLabel?.font = fontBold13
button.backgroundColor = .tunnelsBlue
button.anchors.height.equal(24)
button.layer.cornerRadius = 12
button.anchors.width.equal(100)
button.addTarget(self, action: #selector(upgrade), for: .touchUpInside)
return button
}()
private lazy var protectionPlanLabel: UILabel = {
let label = UILabel()
label.text = NSLocalizedString("Basic Protection", comment: "")
label.font = fontBold22
label.textColor = .label
return label
}()
private lazy var mainTitle: UILabel = {
let label = UILabel()
label.text = NSLocalizedString("Get Anonymous protection", comment: "")
label.font = fontBold24
label.numberOfLines = 0
label.textColor = .label
return label
}()
private lazy var descriptionLabel1: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("Block as many trackers as you want", comment: "")))
return label
}()
private lazy var descriptionLabel2: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("Import and export your own block lists", comment: "")))
return label
}()
private lazy var descriptionLabel3: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("Access to new curated lists of trackers", comment: "")))
return label
}()
private lazy var descriptionLabel4: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("The only fully open source VPN", comment: "")))
return label
}()
private lazy var descriptionLabel5: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("Hide your identity around the world", comment: "")))
return label
}()
private lazy var descriptionLabel6: DescriptionLabel = {
let label = DescriptionLabel()
label.configure(with: DescriptionLabelViewModel(text: NSLocalizedString("Protect all Apple devices", comment: "")))
return label
}()
private lazy var upgradeButton: UIButton = {
let button = UIButton(type: .system)
button.tintColor = .white
button.setTitle(NSLocalizedString("Upgrade", comment: ""), for: .normal)
button.titleLabel?.font = fontBold18
button.backgroundColor = .tunnelsBlue
button.layer.cornerRadius = 28
button.anchors.height.equal(56)
button.addTarget(self, action: #selector(upgrade), for: .touchUpInside)
return button
}()
private lazy var stackView: UIStackView = {
let stackView = UIStackView()
stackView.axis = .vertical
stackView.distribution = .fill
stackView.spacing = 10
stackView.layer.cornerRadius = 8
stackView.backgroundColor = UIColor.dynamicColor(light: .extraLightGray, dark: .panelSecondaryBackground!)
stackView.layoutMargins = UIEdgeInsets(top: 17, left: 20, bottom: 17, right: 20)
stackView.isLayoutMarginsRelativeArrangement = true
return stackView
}()
private lazy var closeButton: UIButton = {
let button = UIButton()
let config = UIImage.SymbolConfiguration(pointSize: 23)
button.setImage(UIImage(systemName: "xmark.circle.fill", withConfiguration: config), for: .normal)
button.tintColor = .secondaryLabel
button.addTarget(self, action: #selector(closeButtonTapped), for: .touchUpInside)
return button
}()
@IBOutlet var mainStack: UIStackView!
@IBOutlet weak var firewallTitleLabel: UILabel!
@IBOutlet weak var firewallActive: UILabel!
@IBOutlet weak var firewallToggleCircle: UIButton!
@IBOutlet weak var firewallToggleAnimatedCircle: NVActivityIndicatorView!
@IBOutlet weak var firewallButton: UIButton!
@IBOutlet weak var tapToActivateFirewallLabel: UILabel!
var lastFirewallStatus: NEVPNStatus?
@IBOutlet weak var metricsStack: UIStackView!
@IBOutlet weak var dailyMetrics: UILabel?
@IBOutlet weak var weeklyMetrics: UILabel?
@IBOutlet weak var allTimeMetrics: UILabel?
var metricsTimer : Timer?
@IBOutlet weak var firewallSettingsButton: UIButton!
@IBOutlet weak var firewallViewLogButton: UIButton!
@IBOutlet weak var firewallShareButton: UIButton!
@IBOutlet weak var vpnHeaderView: UIView!
@IBOutlet weak var vpnActive: UILabel!
@IBOutlet weak var vpnToggleCircle: UIButton!
@IBOutlet weak var vpnToggleAnimatedCircle: NVActivityIndicatorView!
@IBOutlet weak var vpnButton: UIButton!
var lastVPNStatus: NEVPNStatus?
@IBOutlet weak var vpnSetRegionButton: UIButton!
@IBOutlet weak var vpnRegionLabel: UILabel!
@IBOutlet weak var vpnWhitelistButton: UIButton!
private let userService = BaseUserService.shared
var activePlans: [Subscription.PlanType] = []
override func viewDidLoad() {
super.viewDidLoad()
view.addSubview(yourCurrentPlanLabel)
yourCurrentPlanLabel.anchors.leading.marginsPin()
yourCurrentPlanLabel.anchors.top.safeAreaPin(inset: 16)
layoutUI()
updateFirewallButtonWithStatus(status: FirewallController.shared.status())
updateMetrics()
didBecomeActive()
firewallViewLogButton.layer.cornerRadius = 8
firewallViewLogButton.layer.maskedCorners = [.layerMinXMaxYCorner]
firewallSettingsButton.layer.cornerRadius = 8
firewallSettingsButton.layer.maskedCorners = [.layerMaxXMaxYCorner]
updateVPNButtonWithStatus(status: VPNController.shared.status())
//updateIP()
vpnWhitelistButton.layer.cornerRadius = 8
vpnWhitelistButton.layer.maskedCorners = [.layerMinXMaxYCorner]
vpnSetRegionButton.layer.cornerRadius = 8
vpnSetRegionButton.layer.maskedCorners = [.layerMaxXMaxYCorner]
updateVPNRegionLabel()
updateStackViewAxis(basedOn: view.frame.size)
// Check Subscription - if VPN active but not subscribed, then disconnect and show dialog (don't do this if connection error)
if (VPNController.shared.status() == .connected) {
firstly {
try Client.signIn()
}
.done { (signin: SignIn) in
// successfully signed in with no subscription errors, do nothing
}
.catch { error in
if (self.popupErrorAsNSURLError(error)) {
return
}
else if let apiError = error as? ApiError {
switch apiError.code {
case kApiCodeNoSubscriptionInReceipt, kApiCodeNoActiveSubscription:
self.showPopupDialog(title: NSLocalizedString("VPN Subscription Expired", comment: ""), message: NSLocalizedString("Please renew your subscription to re-activate the VPN.", comment: ""), acceptButton: NSLocalizedString("Okay", comment: ""), completionHandler: {
let vc = VPNPaywallViewController()
self.present(vc, animated: true)
// self.performSegue(withIdentifier: "showSignup", sender: self)
})
default:
_ = self.popupErrorAsApiError(error)
}
}
else {
self.showPopupDialog(title: NSLocalizedString("Error Signing In To Verify Subscription", comment: ""),
message: "\(error)",
acceptButton: "Okay")
}
}
}
NotificationCenter.default.addObserver(self, selector: #selector(didBecomeActive), name: UIApplication.didBecomeActiveNotification, object: nil)
NotificationCenter.default.addObserver(self, selector: #selector(willResignActive), name: UIApplication.willResignActiveNotification, object: nil)
}
private func layoutUI() {
if UserDefaults.hasSeenAnonymousPaywall {
mainTitle.text = "Get Universal\nprotection"
protectionPlanLabel.text = "Anonymous protection"
stackView.addArrangedSubview(mainTitle)
stackView.addArrangedSubview(descriptionLabel6)
stackView.addArrangedSubview(upgradeButton)
stackView.setCustomSpacing(14, after: mainTitle)
stackView.setCustomSpacing(14, after: descriptionLabel6)
} else if UserDefaults.hasSeenUniversalPaywall {
protectionPlanLabel.text = "Universal protection"
stackView.anchors.height.equal(0)
contentView.anchors.height.equal(UIScreen.main.bounds.height - 150)
closeButton.isHidden = true
upgradeLabel.isHidden = true
} else if UserDefaults.hasSeenAdvancedPaywall {
mainTitle.text = "Get Anonymous\nprotection"
protectionPlanLabel.text = "Advanced protection"
stackView.addArrangedSubview(mainTitle)
stackView.addArrangedSubview(descriptionLabel4)
stackView.addArrangedSubview(descriptionLabel5)
stackView.setCustomSpacing(14, after: mainTitle)
stackView.setCustomSpacing(14, after: descriptionLabel5)
stackView.addArrangedSubview(upgradeButton)
} else {
mainTitle.text = "Get Advanced\nprotection"
protectionPlanLabel.text = "Basic protection"
stackView.addArrangedSubview(mainTitle)
stackView.addArrangedSubview(descriptionLabel1)
stackView.addArrangedSubview(descriptionLabel2)
stackView.addArrangedSubview(descriptionLabel3)
stackView.addArrangedSubview(upgradeButton)
stackView.setCustomSpacing(14, after: mainTitle)
stackView.setCustomSpacing(14, after: descriptionLabel3)
}
view.addSubview(upgradeLabel)
upgradeLabel.anchors.trailing.marginsPin()
upgradeLabel.anchors.centerY.equal(yourCurrentPlanLabel.anchors.centerY)
view.addSubview(protectionPlanLabel)
protectionPlanLabel.anchors.top.spacing(8, to: yourCurrentPlanLabel.anchors.bottom)
protectionPlanLabel.anchors.leading.marginsPin()
view.addSubview(scrollView)
scrollView.anchors.top.spacing(12, to: protectionPlanLabel.anchors.bottom)
scrollView.anchors.leading.pin()
scrollView.anchors.trailing.pin()
scrollView.anchors.bottom.pin(inset: 80)
scrollView.addSubview(contentView)
contentView.anchors.top.pin()
contentView.anchors.leading.pin()
contentView.anchors.width.equal(scrollView.anchors.width)
contentView.anchors.bottom.pin()
contentView.anchors.trailing.pin()
contentView.addSubview(stackView)
stackView.anchors.top.marginsPin()
stackView.anchors.leading.pin(inset: 22)
stackView.anchors.trailing.pin(inset: 22)
contentView.addSubview(mainStack)
mainStack.anchors.top.spacing(8, to: stackView.anchors.bottom)
mainStack.anchors.leading.marginsPin()
mainStack.anchors.trailing.marginsPin()
mainStack.anchors.bottom.pin()
contentView.addSubview(closeButton)
closeButton.anchors.trailing.spacing(-8, to: stackView.anchors.trailing)
closeButton.anchors.top.marginsPin(inset: 8)
closeButton.anchors.height.equal(40)
closeButton.anchors.width.equal(40)
}
@objc func closeButtonTapped() {
stackView.anchors.height.equal(0)
upgradeButton.isHidden = true
closeButton.isHidden = true
}
@objc func upgrade() {
// showSpecialOffer()
let vc = VPNPaywallViewController()
vc.purchaseSuccessful = {[weak self] in self?.handlePurchaseSuccessful() }
vc.purchaseFailed = { [weak self] err in self?.handlePurchaseFailed(error: err)}
present(vc, animated: true)
}
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
let inset = firewallButton.frame.width * 0.175
firewallButton.contentEdgeInsets = UIEdgeInsets(top: inset, left: inset, bottom: inset, right: inset)
vpnButton.contentEdgeInsets = UIEdgeInsets(top: inset, left: inset, bottom: inset, right: inset)
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
tabBarController?.delegate = self
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
OneTimeActions.performOnce(ifHasNotSeen: .welcomeScreen) {
let vc = WelcomeViewController()
vc.modalPresentationStyle = .overFullScreen
self.present(vc, animated: true)
}
// Used for debugging signup
//performSegue(withIdentifier: "showSignup", sender: nil)
if (defaults.bool(forKey: kHasViewedTutorial) == false) {
// startTutorial()
}
else if (defaults.bool(forKey: kHasSeenEmailSignup) == false) {
AccountUI.presentCreateAccount(on: self)
}
if defaults.bool(forKey: kHasSeenInitialFirewallConnectedDialog) == false {
tapToActivateFirewallLabel.isHidden = false
}
// If total blocked > 1000, and have not shown share dialog before, ask if user wants to share
if (getTotalMetrics() > 1000 && defaults.bool(forKey: kHasSeenShare) != true) {
defaults.set(true, forKey: kHasSeenShare)
let popup = PopupDialog(title: "You've blocked over 1000 trackers! 🎊",
message: NSLocalizedString("Share your anonymized metrics and show other people how to block invasive tracking.", comment: ""),
image: nil,
buttonAlignment: .horizontal,
transitionStyle: .bounceDown,
preferredWidth: 270,
tapGestureDismissal: true,
panGestureDismissal: false,
hideStatusBar: false,
completion: nil)
popup.addButtons([
CancelButton(title: NSLocalizedString("Not Now", comment: ""), dismissOnTap: true) {
let s0 = AwesomeSpotlight(withRect: self.getRectForView(self.firewallShareButton).insetBy(dx: -13.0, dy: -13.0), shape: .roundRectangle, text: NSLocalizedString("You can tap this later if you feel like sharing.\n(Tap anywhere to dismiss)", comment: ""))
let spotlightView = AwesomeSpotlightView(frame: self.view.frame,
spotlight: [s0])
spotlightView.cutoutRadius = 8
spotlightView.spotlightMaskColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.75);
spotlightView.enableArrowDown = true
spotlightView.textLabelFont = fontMedium16
spotlightView.labelSpacing = 24;
spotlightView.delegate = self
self.view.addSubview(spotlightView)
spotlightView.start()
},
DefaultButton(title: NSLocalizedString("Next", comment: ""), dismissOnTap: true) {
self.shareFirewallMetricsTapped("")
}
])
self.present(popup, animated: true, completion: nil)
}
}
@objc
private func didBecomeActive() {
NotificationCenter.default.addObserver(self, selector: #selector(tunnelStatusDidChange(_:)), name: .NEVPNStatusDidChange, object: nil)
startTimer()
let appDelegate = UIApplication.shared.delegate as! AppDelegate
if !defaults.bool(forKey: kOneTimeOfferShown) && appDelegate.timeSinceLastStart > 4 * 60 * 60 {
if BaseUserService.shared.user.currentSubscription == nil {
if let targetDate = Calendar.current.date(from: DateComponents(year: 2024, month: 12, day: 31, hour: 23, minute: 59, second: 59)), Date() < targetDate {
showSpecialOffer()
} else {
showOneTimeOffer()
}
}
}
}
@objc
private func willResignActive() {
NotificationCenter.default.removeObserver(self, name: .NEVPNStatusDidChange, object: nil)
stopTimer()
}
func updateStackViewAxis(basedOn size: CGSize) {
guard traitCollection.userInterfaceIdiom == .pad else {
// axis always vertical on iPhone
return
}
if size.width > size.height {
mainStack.axis = .horizontal
} else {
mainStack.axis = .vertical
}
}
override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
guard traitCollection.userInterfaceIdiom == .pad else {
return
}
coordinator.animate { [unowned self] _ in
self.updateStackViewAxis(basedOn: size)
} completion: { (_) in
return
}
}
private func showOneTimeOffer() {
Task { @MainActor in
try? await Task.sleep(nanoseconds: 1_000_000_000)
if let productInfos = await VPNSubscription.shared.loadSubscriptions(type: .oneTime) {
let model = OneTimePaywallModel(products: VPNSubscription.oneTimeProducts, infos: productInfos)
model.closeAction = { [weak self] in self?.dismiss(animated: true)}
model.continueAction = { [weak self] pid in
VPNSubscription.selectedProductId = pid
VPNSubscription.purchase {
self?.handlePurchaseSuccessful()
} errored: { err in
self?.handlePurchaseFailed(error: err)
}
}
let viewCtrl = UIHostingController(rootView: OneTimePaywallView(model: model))
viewCtrl.modalPresentationStyle = .fullScreen
self.present(viewCtrl, animated: true)
defaults.set(true, forKey: kOneTimeOfferShown)
}
}
}
private func showSpecialOffer() {
Task { @MainActor in
try? await Task.sleep(nanoseconds: 1_000_000_000)
if let productInfos = await VPNSubscription.shared.loadSubscriptions(type: .specialOffer) {
let model = SpecialOfferPaywallModel(products: VPNSubscription.specialOfferProducts, infos: productInfos)
model.closeAction = { [weak self] in self?.dismiss(animated: true)}
model.continueAction = { [weak self] pid in
VPNSubscription.selectedProductId = pid
VPNSubscription.purchase {
self?.handlePurchaseSuccessful()
} errored: { err in
model.showProgress = false
self?.handlePurchaseFailed(error: err)
}
}
let viewCtrl = UIHostingController(rootView: SpecialOfferPaywallView(model: model))
viewCtrl.modalPresentationStyle = .fullScreen
self.present(viewCtrl, animated: true)
defaults.set(true, forKey: kOneTimeOfferShown)
}
}
}
func showVPNSubscriptionDialog(title: String, message: String) {
let popup = PopupDialog(
title: title,
message: message,
image: nil,
buttonAlignment: .horizontal,
transitionStyle: .bounceUp,
preferredWidth: 300.0,
tapGestureDismissal: false,
panGestureDismissal: false,
hideStatusBar: true,
completion: nil)
let getEnhancedPrivacyButton = DefaultButton(title: NSLocalizedString("1 Week Free", comment: ""), dismissOnTap: true) { [unowned self] in
let vc = VPNPaywallViewController()
present(vc, animated: true)
}
let laterButton = CancelButton(title: NSLocalizedString("Skip Trial", comment: ""), dismissOnTap: true) { }
popup.addButtons([laterButton, getEnhancedPrivacyButton])
self.present(popup, animated: true, completion: nil)
}
// This notification is triggered for both Firewall and VPN
@objc func tunnelStatusDidChange(_ notification: Notification) {
// Firewall
if let tunnelProviderSession = notification.object as? NETunnelProviderSession {
DDLogInfo("VPNStatusDidChange as NETunnelProviderSession with status: \(tunnelProviderSession.status.description)");
updateFirewallButtonWithStatus(status: tunnelProviderSession.status)
if getUserWantsFirewallEnabled() &&
tunnelProviderSession.status == .connected &&
defaults.bool(forKey: kHasSeenInitialFirewallConnectedDialog) == false {
defaults.set(true, forKey: kHasSeenInitialFirewallConnectedDialog)
self.tapToActivateFirewallLabel.isHidden = true
let hasSubscription = UserDefaults.hasSeenAnonymousPaywall || UserDefaults.hasSeenUniversalPaywall || UserDefaults.hasSeenAdvancedPaywall
if (VPNController.shared.status() == .invalid && !hasSubscription) {
self.showVPNSubscriptionDialog(title: NSLocalizedString("🔥🧱 Firewall Activated 🎊🎉", comment: ""), message: NSLocalizedString("Trackers, ads, and other malicious scripts are now blocked in all your apps, even outside of Safari.\n\nGet maximum privacy with a Secure Tunnel that protects connections, anonymizes your browsing, and hides your location.", comment: ""))
}
}
}
// VPN
else if let neVPNConnection = notification.object as? NEVPNConnection {
DDLogInfo("VPNStatusDidChange as NEVPNConnection with status: \(neVPNConnection.status.description)");
updateVPNButtonWithStatus(status: neVPNConnection.status);
updateVPNRegionLabel()
if NEVPNManager.shared().connection.status == .connected || NEVPNManager.shared().connection.status == .disconnected {
//self.updateIP();
}
}
else {
DDLogInfo("VPNStatusDidChange neither TunnelProviderSession nor NEVPNConnection");
}
}
// MARK: - Top Buttons
func startTutorial() {
var spotlights: [AwesomeSpotlight] = []
let centerPoint = UIScreen.main.bounds.center
let s0 = AwesomeSpotlight(withRect: CGRect(x: centerPoint.x, y: centerPoint.y - 100, width: 0, height: 0), shape: .circle, text: NSLocalizedString("Welcome to the Lockdown Tutorial.\n\nTap anywhere to continue.", comment: ""))
let s1 = AwesomeSpotlight(withRect: getRectForView(firewallTitleLabel).insetBy(dx: -13.0, dy: -13.0), shape: .roundRectangle, text: NSLocalizedString("Lockdown Firewall blocks bad and untrusted connections in all your apps - not just Safari.", comment: ""))
let s2 = AwesomeSpotlight(withRect: getRectForView(firewallToggleCircle).insetBy(dx: -10.0, dy: -10.0), shape: .circle, text: NSLocalizedString("Activate Firewall with this button.", comment: ""))
let s3 = AwesomeSpotlight(withRect: getRectForView(metricsStack).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("See live metrics for how many bad connections Firewall has blocked.", comment: ""))
let s4 = AwesomeSpotlight(withRect: getRectForView(firewallViewLogButton).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("\"View Log\" shows exactly what connections were blocked in the past day. This log is cleared at midnight and stays on-device, so it's only visible to you.", comment: ""))
let s5 = AwesomeSpotlight(withRect: getRectForView(firewallSettingsButton).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("\"Block List\" lets you choose what you want to block (e.g, Facebook, clickbait, etc). You can also set custom domains to block.", comment: ""))
let s6 = AwesomeSpotlight(withRect: getRectForView(vpnHeaderView).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("For maximum privacy, activate Secure Tunnel, which uses bank-level encryption to protect connections, anonymize your browsing, and hide your location and IP.", comment: ""))
spotlights.append(contentsOf: [s0, s1, s2, s3, s4, s5, s6])
if let tabBarButton = (tabBarController as? MainTabBarController)?.accountTabBarButton {
let s7 = AwesomeSpotlight(withRect: getRectForView(tabBarButton).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("To see this tutorial again, open the Account tab.", comment: ""))
spotlights.append(s7)
}
let spotlightView = AwesomeSpotlightView(frame: view.frame,
spotlight: spotlights)
spotlightView.accessibilityIdentifier = "tutorial"
spotlightView.cutoutRadius = 8
spotlightView.spotlightMaskColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.75);
spotlightView.enableArrowDown = true
spotlightView.textLabelFont = fontMedium16
spotlightView.labelSpacing = 24;
spotlightView.delegate = self
tabBarController?.view.addSubview(spotlightView)
spotlightView.start()
}
func spotlightViewDidCleanup(_ spotlightView: AwesomeSpotlightView) {
guard spotlightView.accessibilityIdentifier == "tutorial" else {
return
}
defaults.set(true, forKey: kHasViewedTutorial)
if getAPICredentials() != nil {
// already has email signup pending or confirmed, don't show create account
}
else {
AccountUI.presentCreateAccount(on: self)
}
}
@IBAction func shareFirewallMetricsTapped(_ sender: Any) {
let thousandsFormatter = NumberFormatter()
thousandsFormatter.groupingSeparator = ","
thousandsFormatter.numberStyle = .decimal
let imageSize = CGSize(width: 720, height: 420)
let renderer = UIGraphicsImageRenderer(size: imageSize)
let image = renderer.image { ctx in
let rectangle = CGRect(origin: CGPoint.zero, size: imageSize)
ctx.cgContext.setFillColor(UIColor.white.cgColor)
ctx.cgContext.addRect(rectangle)
ctx.cgContext.drawPath(using: .fill)
UIImage(named: "share.png")!.draw(in: CGRect(origin: CGPoint.zero, size: imageSize))
let paragraphStyle = NSMutableParagraphStyle()
paragraphStyle.alignment = .center
let sinceAttrs = [NSAttributedString.Key.font: UIFont.systemFont(ofSize: 30, weight: .semibold), NSAttributedString.Key.paragraphStyle: paragraphStyle, NSAttributedString.Key.foregroundColor: UIColor(red: 176/255, green: 176/255, blue: 176/255, alpha: 0.59)]
let sinceY = 90
var date = "INSTALL"
let formatter = DateFormatter()
formatter.dateFormat = "MMM d YYYY"
if let appInstall = appInstallDate {
date = formatter.string(from: appInstall).uppercased()
}
"SINCE \(date)".draw(with: CGRect(origin: CGPoint(x: 0, y: sinceY), size: CGSize(width: 720, height: 50)), options: .usesLineFragmentOrigin, attributes: sinceAttrs, context: nil)
let attrs = [NSAttributedString.Key.font: UIFont.systemFont(ofSize: 46, weight: .bold), NSAttributedString.Key.paragraphStyle: paragraphStyle, NSAttributedString.Key.foregroundColor: UIColor(red: 149/255, green: 149/255, blue: 149/255, alpha: 1.0)]
let countSize = CGSize(width: 240, height: 50)
let countY = 216
thousandsFormatter.string(for: getDayMetrics())!.draw(with: CGRect(origin: CGPoint(x: 0, y: countY), size: countSize), options: .usesLineFragmentOrigin, attributes: attrs, context: nil)
thousandsFormatter.string(for: getWeekMetrics())!.draw(with: CGRect(origin: CGPoint(x: 240, y: countY), size: countSize), options: .usesLineFragmentOrigin, attributes: attrs, context: nil)
thousandsFormatter.string(for: getTotalMetrics())!.draw(with: CGRect(origin: CGPoint(x: 480, y: countY), size: countSize), options: .usesLineFragmentOrigin, attributes: attrs, context: nil)
}
let popup = PopupDialog(
title: NSLocalizedString("Share Your Stats", comment: ""),
message: NSLocalizedString("Show how invasive today's apps are, and help other people block trackers and badware, too.\n\nYour block log is not included - only the image above. Choose where to share in the next step.", comment: ""),
image: image,
buttonAlignment: .horizontal,
transitionStyle: .bounceDown,
preferredWidth: 300.0,
tapGestureDismissal: true,
panGestureDismissal: false,
hideStatusBar: true,
completion: nil)
let cancelButton = CancelButton(title: NSLocalizedString("Cancel", comment: ""), dismissOnTap: true) { }
let shareButton = DefaultButton(title: NSLocalizedString("Next", comment: ""), dismissOnTap: true) {
let shareText = "\(NSLocalizedString("I blocked", comment: "Used in the sentence: I blocked 500 trackers with Lockdown.")) \(thousandsFormatter.string(for: getTotalMetrics())!)\(NSLocalizedString(" trackers, ads, and badware with Lockdown, the firewall that blocks unwanted connections in all your apps. Get it free at lockdownprivacy.com.", comment: "Used in the sentence: I blocked 500 trackers, ads, and badware with Lockdown, the firewall that blocks unwanted connections in all your apps. Get it free at lockdownprivacy.com."))"
let vc = UIActivityViewController(activityItems: [LockdownCustomActivityItemProvider(text: shareText), image], applicationActivities: [])
vc.completionWithItemsHandler = { (activity, success, items, error) in
if (success) {
self.showPopupDialog(title: NSLocalizedString("Success!", comment: ""), message: NSLocalizedString("Thanks for helping to increase privacy and tracking awareness.", comment: ""), acceptButton: NSLocalizedString("Nice", comment: "Used as a button text in a popup. Like 'OK' except more excited."))
}
}
vc.excludedActivityTypes = [ UIActivity.ActivityType.assignToContact, UIActivity.ActivityType.addToReadingList, UIActivity.ActivityType.openInIBooks, UIActivity.ActivityType.postToVimeo, UIActivity.ActivityType.print ]
if let popoverPC = vc.popoverPresentationController {
popoverPC.sourceView = self.firewallShareButton
popoverPC.sourceRect = self.firewallShareButton.bounds
popoverPC.permittedArrowDirections = .up
}
self.present(vc, animated: true)
}
popup.addButtons([cancelButton, shareButton])
self.present(popup, animated: true, completion: nil)
}
// MARK: - Firewall
private func startTimer() {
stopTimer()
metricsTimer = Timer.scheduledTimer(
timeInterval: 2.0,
target: self,
selector: #selector(updateMetrics),
userInfo: nil,
repeats: true
)
metricsTimer?.fire()
}
private func stopTimer() {
metricsTimer?.invalidate()
metricsTimer = nil
}
@objc func updateMetrics() {
DispatchQueue.main.async {
self.dailyMetrics?.text = getDayMetricsString()
self.weeklyMetrics?.text = getWeekMetricsString()
self.allTimeMetrics?.text = getTotalMetricsString()
}
}
@IBAction func toggleFirewall(_ sender: Any) {
if (defaults.bool(forKey: kHasAgreedToFirewallPrivacyPolicy) == false) {
let storyboard = UIStoryboard(name: "Main", bundle: nil)
let viewController = storyboard.instantiateViewController(withIdentifier: "firewallPrivacyPolicyViewController") as! PrivacyPolicyViewController
viewController.privacyPolicyKey = kHasAgreedToFirewallPrivacyPolicy
viewController.parentVC = self
self.present(viewController, animated: true, completion: nil)
return
}
if getIsCombinedBlockListEmpty() {
FirewallController.shared.setEnabled(false, isUserExplicitToggle: true)
self.showPopupDialog(title: NSLocalizedString("No Block Lists Enabled", comment: ""), message: NSLocalizedString("Please tap Block List and enable at least one block list to activate Firewall.", comment: ""), acceptButton: NSLocalizedString("Okay", comment: ""))
return
}
switch FirewallController.shared.status() {
case .invalid:
FirewallController.shared.setEnabled(true, isUserExplicitToggle: true)
case .disconnected:
ensureFirewallWorkingAfterEnabling(waitingSeconds: 3.0)
updateFirewallButtonWithStatus(status: .connecting)
FirewallController.shared.setEnabled(true, isUserExplicitToggle: true)
checkForAskRating()
case .connected:
updateFirewallButtonWithStatus(status: .disconnecting)
FirewallController.shared.setEnabled(false, isUserExplicitToggle: true)
case .connecting, .disconnecting, .reasserting:
ensureFirewallWorkingAfterEnabling(waitingSeconds: 3.0)
break;
}
}
func ensureFirewallWorkingAfterEnabling(waitingSeconds: TimeInterval) {
FirewallController.shared.existingManagerCount { (count) in
if let count = count, count > 0 {
DispatchQueue.main.asyncAfter(deadline: .now() + waitingSeconds) {
DDLogInfo("\(waitingSeconds) seconds passed, checking if Firewall is enabled")
guard getUserWantsFirewallEnabled() else {
// firewall shouldn't be enabled, no need to act
DDLogInfo("User doesn't want Firewall enabled, no action")
return
}
let status = FirewallController.shared.status()
switch status {
case .connecting, .disconnecting, .reasserting:
// check again in three seconds
DDLogInfo("Firewall is in transient state, will check again in 3 seconds")
self.ensureFirewallWorkingAfterEnabling(waitingSeconds: 3.0)
case .connected:
// all good
DDLogInfo("Firewall is connected, no action")
break
case .disconnected, .invalid:
// we suppose that the connection is somehow broken, trying to fix
DDLogInfo("Firewall is not connected even though it should be, attempting to fix")
self.showFixFirewallConnectionDialog {
FirewallController.shared.deleteConfigurationAndAddAgain()
}
}
}
} else {
DDLogInfo("No Firewall configurations in settings (likely fresh install): not checking")
return
}
}
}
func updateFirewallButtonWithStatus(status: NEVPNStatus) {
DDLogInfo("UpdateFirewallButton")
switch status {
case .connected:
LatestKnowledge.isFirewallEnabled = true
case .disconnected:
LatestKnowledge.isFirewallEnabled = false
default:
break
}
updateToggleButtonWithStatus(lastStatus: lastFirewallStatus,
newStatus: status,
activeLabel: firewallActive,
toggleCircle: firewallToggleCircle,
toggleAnimatedCircle: firewallToggleAnimatedCircle,
button: firewallButton,
prefixText: NSLocalizedString("Firewall", comment: "").uppercased())
}
func updateToggleButtonWithStatus(lastStatus: NEVPNStatus?, newStatus: NEVPNStatus, activeLabel: UILabel, toggleCircle: UIButton, toggleAnimatedCircle: NVActivityIndicatorView, button: UIButton, prefixText: String) {
DDLogInfo("UpdateToggleButton")
if (newStatus == lastStatus) {
DDLogInfo("No status change from last time, ignoring.");
}
else {
DispatchQueue.main.async() {
switch newStatus {
case .connected:
activeLabel.text = "\(prefixText)\(NSLocalizedString(" On", comment: "").uppercased())"
activeLabel.backgroundColor = UIColor.tunnelsBlue
toggleCircle.tintColor = .tunnelsBlue
toggleCircle.isHidden = false
toggleAnimatedCircle.stopAnimating()
button.tintColor = .tunnelsBlue
case .connecting:
activeLabel.text = NSLocalizedString("Activating", comment: "").uppercased()
activeLabel.backgroundColor = .tunnelsBlue
toggleCircle.isHidden = true
toggleAnimatedCircle.color = .tunnelsBlue
toggleAnimatedCircle.startAnimating()
button.tintColor = .tunnelsBlue
case .disconnected, .invalid:
activeLabel.text = "\(prefixText)\(NSLocalizedString(" Off", comment: "").uppercased())"
activeLabel.backgroundColor = .tunnelsWarning
toggleCircle.tintColor = .lightGray
toggleCircle.isHidden = false
toggleAnimatedCircle.stopAnimating()
button.tintColor = .lightGray
case .disconnecting:
activeLabel.text = NSLocalizedString("Deactivating", comment: "").uppercased()
activeLabel.backgroundColor = .lightGray
toggleCircle.isHidden = true
toggleAnimatedCircle.color = .lightGray
toggleAnimatedCircle.startAnimating()
button.tintColor = .lightGray
case .reasserting:
break;
}
}
}
}
func highlightBlockLog() {
let blockLogSpotlight = AwesomeSpotlight(withRect: getRectForView(firewallViewLogButton).insetBy(dx: -10.0, dy: -10.0), shape: .roundRectangle, text: NSLocalizedString("Tap to see the blocked tracking attempts.", comment: ""))
let spotlightView = AwesomeSpotlightView(frame: view.frame, spotlight: [blockLogSpotlight])
spotlightView.accessibilityIdentifier = "highlightBlockLog"
spotlightView.cutoutRadius = 8
spotlightView.spotlightMaskColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.75);
spotlightView.enableArrowDown = true
spotlightView.textLabelFont = fontMedium16
spotlightView.labelSpacing = 24;
view.addSubview(spotlightView)
spotlightView.start()
}
// MARK: - VPN
@IBAction func vpnQuestionTapped(_ sender: Any) {
self.performSegue(withIdentifier: "showWhatIsVPN", sender: self)
}
func updateVPNButtonWithStatus(status: NEVPNStatus) {
DDLogInfo("UpdateVPNButton")
switch status {
case .connected:
LatestKnowledge.isVPNEnabled = true
case .disconnected:
LatestKnowledge.isVPNEnabled = false
default:
break
}
updateToggleButtonWithStatus(lastStatus: lastVPNStatus,
newStatus: status,
activeLabel: vpnActive,
toggleCircle: vpnToggleCircle,
toggleAnimatedCircle: vpnToggleAnimatedCircle,
button: vpnButton,
prefixText: NSLocalizedString("Tunnel", comment: "").uppercased())
}
private func startVPNAfterSettingCredentials() {
VPNController.shared.setEnabled(true) { error in
if error != nil {
self.updateVPNButtonWithStatus(status: .disconnected)
}
}
}
@IBAction func toggleVPN(_ sender: Any) {
if UserDefaults.hasSeenAnonymousPaywall || UserDefaults.hasSeenUniversalPaywall {
DDLogInfo("Toggle VPN")
switch VPNController.shared.status() {
case .connected, .connecting, .reasserting:
DDLogInfo("Toggle VPN: on currently, turning it off")
updateVPNButtonWithStatus(status: .disconnecting)
VPNController.shared.setEnabled(false)
case .disconnected, .disconnecting, .invalid:
DDLogInfo("Toggle VPN: off currently, turning it on")
updateVPNButtonWithStatus(status: .connecting)
// VPNController.shared.setEnabled(true)
// ensureFirewallWorkingAfterEnabling(waitingSeconds: 5.0)
// if there's a confirmed email, use that and sync the receipt with it
if let apiCredentials = getAPICredentials(), getAPICredentialsConfirmed() == true {
DDLogInfo("have confirmed API credentials, using them")
firstly {
try Client.signInWithEmail(email: apiCredentials.email, password: apiCredentials.password)
}
.then { (signin: SignIn) -> Promise<SubscriptionEvent> in
DDLogInfo("signin result: \(signin)")
return try Client.subscriptionEvent()
}
.then { (result: SubscriptionEvent) -> Promise<GetKey> in
DDLogInfo("subscriptionevent result: \(result)")
return try Client.getKey()
}
.done { [weak self] (getKey: GetKey) in
try setVPNCredentials(id: getKey.id, keyBase64: getKey.b64)
DDLogInfo("setting VPN creds with ID: \(getKey.id)")
self?.startVPNAfterSettingCredentials()
}
.catch { error in
DDLogError("Error doing email-login -> subscription-event: \(error)")
self.updateVPNButtonWithStatus(status: .disconnected)
if (self.popupErrorAsNSURLError(error)) {
return
}
else if let apiError = error as? ApiError {
switch apiError.code {
case kApiCodeInvalidAuth, kApiCodeIncorrectLogin:
let confirm = PopupDialog(title: "Incorrect Login",
message: "Your saved login credentials are incorrect. Please sign out and try again.",
image: nil,
buttonAlignment: .horizontal,
transitionStyle: .bounceDown,
preferredWidth: 270,
tapGestureDismissal: true,
panGestureDismissal: false,
hideStatusBar: false,
completion: nil)
confirm.addButtons([
DefaultButton(title: NSLocalizedString("Cancel", comment: ""), dismissOnTap: true) {
},
DefaultButton(title: NSLocalizedString("Sign Out", comment: ""), dismissOnTap: true) {
URLCache.shared.removeAllCachedResponses()
Client.clearCookies()
clearAPICredentials()
setAPICredentialsConfirmed(confirmed: false)
self.showPopupDialog(title: "Success", message: "Signed out successfully.", acceptButton: NSLocalizedString("Okay", comment: ""))
},
])
self.present(confirm, animated: true, completion: nil)
case kApiCodeNoSubscriptionInReceipt:
self.performSegue(withIdentifier: "showSignup", sender: self)
case kApiCodeNoActiveSubscription:
self.showPopupDialog(title: NSLocalizedString("Subscription Expired", comment: ""), message: NSLocalizedString("Please renew your subscription to activate the Secure Tunnel.", comment: ""), acceptButton: NSLocalizedString("Okay", comment: ""), completionHandler: {
self.performSegue(withIdentifier: "showSignup", sender: self)
})
default:
_ = self.popupErrorAsApiError(error)
}
}
}
}
else {
firstly {
try Client.signIn() // this will fetch and set latest receipt, then submit to API to get cookie
}
.then { (signin: SignIn) -> Promise<GetKey> in
// TODO: don't always do this -- if we already have a key, then only do it once per day max
try Client.getKey()
}
.done { [weak self] (getKey: GetKey) in
try setVPNCredentials(id: getKey.id, keyBase64: getKey.b64)
self?.startVPNAfterSettingCredentials()
}
.catch { error in
self.updateVPNButtonWithStatus(status: .disconnected)
if (self.popupErrorAsNSURLError(error)) {
return
}
else if let apiError = error as? ApiError {
switch apiError.code {
case kApiCodeNoSubscriptionInReceipt:
self.performSegue(withIdentifier: "showSignup", sender: self)
case kApiCodeNoActiveSubscription:
self.showPopupDialog(title: NSLocalizedString("Subscription Expired", comment: ""), message: NSLocalizedString("Please renew your subscription to activate the Secure Tunnel.", comment: ""), acceptButton: NSLocalizedString("Okay", comment: ""), completionHandler: {
self.performSegue(withIdentifier: "showSignup", sender: self)
})
default:
if (apiError.code == kApiCodeNegativeError) {
if (getVPNCredentials() != nil) {
DDLogError("Unknown error -1 from API, but VPNCredentials exists, so activating anyway.")
self.updateVPNButtonWithStatus(status: .connecting)
self.startVPNAfterSettingCredentials()
}
else {
self.showPopupDialog(title: NSLocalizedString("Apple Outage", comment: ""), message: "There is currently an outage at Apple which is preventing Secure Tunnel from activating. This will likely by resolved by Apple soon, and we apologize for this issue in the meantime." + NSLocalizedString("\n\n If this error persists, please contact team@lockdownprivacy.com.", comment: ""), acceptButton: NSLocalizedString("Okay", comment: ""))
}
}
else {
_ = self.popupErrorAsApiError(error)
}
}
}
else {
self.showPopupDialog(title: NSLocalizedString("Error Signing In To Verify Subscription", comment: ""),
message: "\(error)",
acceptButton: NSLocalizedString("Okay", comment: ""))
}
}
}
}
} else { upgrade() }
}
@IBAction func viewAuditReportTapped(_ sender: Any) {
showAuditModal()
}
@IBAction func showWhitelist(_ sender: Any) {
if UserDefaults.hasSeenUniversalPaywall || UserDefaults.hasSeenAnonymousPaywall {
performSegue(withIdentifier: "showWhitelist", sender: nil)
} else {
upgrade()
}
}
@IBAction func showSetRegion(_ sender: Any) {
if UserDefaults.hasSeenUniversalPaywall || UserDefaults.hasSeenAnonymousPaywall {
performSegue(withIdentifier: "showSetRegion", sender: nil)
} else {
upgrade()
}
}
func showBlockLog(_ sender: Any) {
performSegue(withIdentifier: "showBlockLog", sender: nil)
}
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
switch segue.identifier {
case "showSetRegion":
if let vc = segue.destination as? SetRegionViewController {
vc.homeVC = self
}
// case "showWhatIsVPN":
// if let vc = segue.destination as? WhatIsVpnViewController {
// vc.parentVC = self
// }
default:
break
}
}
func handlePurchaseFailed(error: Error) {
// let keyWindow = UIApplication.shared.windows.first(where: { $0.isKeyWindow })
// let vc = SplashscreenViewController()
// let navigation = UINavigationController(rootViewController: vc)
// keyWindow?.rootViewController = navigation
// DDLogError("Start Trial Failed: \(error)")
if let skError = error as? SKError {
var errorText = ""
switch skError.code {
case .unknown:
errorText = .localized("Unknown error. Please contact support at team@lockdownprivacy.com.")
case .clientInvalid:
errorText = .localized("Not allowed to make the payment")
case .paymentCancelled:
errorText = .localized("Payment was cancelled")
case .paymentInvalid:
errorText = .localized("The purchase identifier was invalid")
case .paymentNotAllowed:
errorText = .localized("""
Payment not allowed.\nEither this device is not allowed to make purchases, or In-App Purchases have been disabled. \
Please allow them in Settings App -> Screen Time -> Restrictions -> App Store -> In-app Purchases. Then try again.
""")
case .storeProductNotAvailable:
errorText = .localized("The product is not available in the current storefront")
case .cloudServicePermissionDenied:
errorText = .localized("Access to cloud service information is not allowed")
case .cloudServiceNetworkConnectionFailed:
errorText = .localized("Could not connect to the network")
case .cloudServiceRevoked:
errorText = .localized("User has revoked permission to use this cloud service")
default:
errorText = skError.localizedDescription
}
self.showPopupDialog(title: .localized("Error Making Purchase"), message: errorText, acceptButton: .localizedOkay)
}
else if self.popupErrorAsNSURLError(error) {
return
}
else if self.popupErrorAsApiError(error) {
return
}
else {
self.showPopupDialog(
title: .localized("Error Making Purchase"),
message: .localized("Please contact team@lockdownprivacy.com.\n\nError details:\n") + "\(error)",
acceptButton: .localizedOkay)
}
}
func handlePurchaseSuccessful(completion: (()->Void)? = nil) {
let keyWindow = UIApplication.shared.windows.first(where: { $0.isKeyWindow })
let vc = SplashscreenViewController()
let navigation = UINavigationController(rootViewController: vc)
keyWindow?.rootViewController = navigation
// force refresh receipt, and sync with email if it exists
if let apiCredentials = getAPICredentials(), getAPICredentialsConfirmed() == true {
DDLogInfo("purchase complete: syncing with confirmed email")
firstly {
try Client.signInWithEmail(email: apiCredentials.email, password: apiCredentials.password)
}
.then { (signin: SignIn) -> Promise<SubscriptionEvent> in
DDLogInfo("purchase complete: signin result: \(signin)")
return try Client.subscriptionEvent(forceRefresh: true)
}
.then { (result: SubscriptionEvent) -> Promise<[Subscription]> in
DDLogInfo("plan status: subscriptionevent result: \(result)")
return try Client.activeSubscriptions()
}
.done { subscriptions in
DDLogInfo("active-subs (start trial): \(subscriptions)")
NotificationCenter.default.post(name: AccountUI.accountStateDidChange, object: self)
self.userService.user.updateSubscription(to: subscriptions.first)
}
.ensure {
completion?()
}
.catch { error in
DDLogError("purchase complete: Error: \(error)")
if self.popupErrorAsNSURLError("Error activating Secure Tunnel: \(error)") {
return
} else if let apiError = error as? ApiError {
switch apiError.code {
default:
_ = self.popupErrorAsApiError("API Error activating Secure Tunnel: \(error)")
}
}
}
} else {
firstly {
try Client.signIn()
}.then { _ in
try Client.activeSubscriptions()
}.done { subscriptions in
DDLogInfo("active-subs (start trial): \(subscriptions)")
NotificationCenter.default.post(name: AccountUI.accountStateDidChange, object: self)
self.userService.user.updateSubscription(to: subscriptions.first)
}
.catch { error in
DDLogError("purchase complete - no email: Error: \(error)")
if self.popupErrorAsNSURLError("Error activating Secure Tunnel: \(error)") {
return
} else if let apiError = error as? ApiError {
switch apiError.code {
default:
_ = self.popupErrorAsApiError("API Error activating Secure Tunnel: \(error)")
}
}
}
}
}
func updateVPNRegionLabel() {
vpnRegionLabel.text = getSavedVPNRegion().regionDisplayNameShort
}
// MARK: - Helpers
func checkForAskRating(delayInSeconds: TimeInterval = 5.0) {
DDLogInfo("Checking for ask rating")
let ratingCount = defaults.integer(forKey: ratingCountKey) + 1
DDLogInfo("Incrementing Rating Count: " + String(ratingCount))
defaults.set(ratingCount, forKey: ratingCountKey)
// not testflight
if (isTestFlight) {
DDLogInfo("Not doing rating for TestFlight")
return
}
// greater than 3 days since install
if let installDate = appInstallDate, let daysSinceInstall = Calendar.current.dateComponents([.day], from: installDate, to: Date()).day, daysSinceInstall <= 3 {
DDLogInfo("Rating Check: Skipping - App was installed on \(installDate), fewer than 4 days since install - \(daysSinceInstall) days")
return
}
// only check every 8th time connecting to this version
if (ratingCount % 8 != 0) {
DDLogInfo("Rating Check: Skipping - ratingCount % 8 != 0: \(ratingCount)")
return
}
// hasn't asked for this version 2 times already
let ratingTriggered = defaults.integer(forKey: ratingTriggeredKey)
if (ratingTriggered >= 3) {
DDLogInfo("Rating Check: Skipping - ratingTriggered greater or equal to 3: \(ratingTriggered)")
return
}
// passed all checks, ask for rating
DispatchQueue.main.asyncAfter(deadline: .now() + delayInSeconds) {
defaults.set(ratingTriggered + 1, forKey: self.ratingTriggeredKey)
SKStoreReviewController.requestReview()
}
}
}
// MARK: - Paywalling
extension HomeViewController: PaywallViewControllerCloseDelegate {
func didClosePaywall() {
BaseUserService.shared.updateUserSubscription { [weak self] subscription in
self?.showLoadingView()
DispatchQueue.main.async {
if subscription?.planType == .anonymousMonthly || subscription?.planType == .anonymousAnnual {
UserDefaults.hasSeenAdvancedPaywall = false
UserDefaults.hasSeenUniversalPaywall = false
UserDefaults.hasSeenAnonymousPaywall = true
}
else if subscription?.planType == .universalMonthly || subscription?.planType == .universalAnnual {
UserDefaults.hasSeenAnonymousPaywall = false
UserDefaults.hasSeenAdvancedPaywall = false
UserDefaults.hasSeenUniversalPaywall = true
}
else if subscription?.planType == .advancedMonthly || subscription?.planType == .advancedAnnual {
UserDefaults.hasSeenAnonymousPaywall = false
UserDefaults.hasSeenAdvancedPaywall = true
UserDefaults.hasSeenUniversalPaywall = false
}
else {
UserDefaults.hasSeenAnonymousPaywall = false
UserDefaults.hasSeenAdvancedPaywall = false
UserDefaults.hasSeenUniversalPaywall = false
}
}
self?.hideLoadingView()
}
}
private func showEnableNotifications() {
let enableNotificationsViewController = EnableNotificationsViewController()
enableNotificationsViewController.modalPresentationStyle = .overFullScreen
present(enableNotificationsViewController, animated: true)
}
private func showPaywallIfNoSubscription() {
guard BaseUserService.shared.user.currentSubscription == nil else { return }
guard BasePaywallService.shared.context == .normal else { return }
BasePaywallService.shared.showPaywall(on: self)
UserDefaults.hasSeenPaywallOnHomeScreen = true
}
}
class LockdownCustomActivityItemProvider : UIActivityItemProvider {
let shareText: String
init(text: String) {
self.shareText = text
super.init(placeholderItem: text)
}
override func activityViewController(_ activityViewController: UIActivityViewController, itemForActivityType activityType: UIActivity.ActivityType?) -> Any? {
if let type = activityType {
switch type {
case UIActivity.ActivityType.postToTwitter:
return shareText + " @lockdown_hq"
default:
return shareText
}
}
else {
return shareText
}
}
}
fileprivate extension PopupDialogButton {
func startActivityIndicator() {
let activity = UIActivityIndicatorView()
if let label = titleLabel {
label.addSubview(activity)
activity.translatesAutoresizingMaskIntoConstraints = false
activity.centerYAnchor.constraint(equalTo: label.centerYAnchor).isActive = true
activity.leadingAnchor.constraint(equalToSystemSpacingAfter: label.trailingAnchor, multiplier: 1).isActive = true
activity.startAnimating()
}
}
func stopActivityIndicator() {
if let label = titleLabel {
let indicators = label.subviews.compactMap { $0 as? UIActivityIndicatorView }
for indicator in indicators {
indicator.stopAnimating()
indicator.removeFromSuperview()
}
}
}
}
final class DynamicButton: PopupDialogButton {
var onTap: ((DynamicButton) -> ())?
override var buttonAction: PopupDialogButton.PopupDialogButtonAction? {
get {
if let onTap = onTap {
return { [weak self] in if let value = self { return onTap(value) } }
} else {
return nil
}
}
}
}
extension NEVPNStatus: CustomStringConvertible {
public var description: String {
switch self {
case .invalid:
return "invalid"
case .disconnected:
return "disconnected"
case .connecting:
return "connecting"
case .connected:
return "connected"
case .reasserting:
return "reasserting"
case .disconnecting:
return "disconnecting"
}
}
}
extension HomeViewController: PurchaseHandler {
func purchase(productId: String) {
VPNSubscription.selectedProductId = productId
VPNSubscription.purchase { [weak self] in
self?.handlePurchaseSuccessful()
} errored: { [weak self] err in
self?.handlePurchaseFailed(error: err)
}
}
}
extension HomeViewController: UITabBarControllerDelegate {
func tabBarController(_ tabBarController: UITabBarController, shouldSelect viewController: UIViewController) -> Bool {
if viewController is UINavigationController || viewController is HomeViewController {
return true
}
feedbackFlow?.startFlow()
return false
}
func showFeedbackPaywall() async {
guard let productInfos = await VPNSubscription.shared.loadSubscriptions(type: .feedback) else { return }
let viewModel = FeedbackPaywallViewModel(products: VPNSubscription.feedbackProducts, subscriptionInfo: productInfos)
viewModel.onCloseHandler = { vc in vc.dismiss(animated: true) }
viewModel.onPurchaseHandler = { [weak self] paywallVC, pid in
self?.purchase(productId: pid)
}
let paywalVC = FeedbackPaywallViewController(viewModel: viewModel)
present(paywalVC, animated: true)
}
}