Files
lockdown-iOS-mirror/LockdowniOS/HomeViewController.swift

881 lines
43 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 SwiftyStoreKit
import PopupDialog
import AwesomeSpotlightView
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
@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!
var activePlans: [Subscription.PlanType] = []
override func viewDidLoad() {
super.viewDidLoad()
updateFirewallButtonWithStatus(status: FirewallController.shared.status())
updateMetrics()
if metricsTimer == nil {
metricsTimer = Timer.scheduledTimer(timeInterval: 2.0, target: self, selector: #selector(updateMetrics), userInfo: nil, repeats: true)
metricsTimer?.fire()
}
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: {
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(tunnelStatusDidChange(_:)), name: .NEVPNStatusDidChange, object: nil)
}
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 viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
// 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
}
OneTimeActions.performOnce(ifHasNotSeen: .notificationAuthorizationRequestPopup) {
PushNotifications.Authorization.requestWeeklyUpdateAuthorization(presentingDialogOn: self).done { status in
DDLogInfo("Updated notifications status: \(status)")
}.catch { error in
DDLogWarn(error.localizedDescription)
}
}
// 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)
}
}
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
}
}
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) {
self.performSegue(withIdentifier: "showSignup", sender: self)
}
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)");
if (!getUserWantsFirewallEnabled()) {
updateFirewallButtonWithStatus(status: .disconnected)
}
else {
updateFirewallButtonWithStatus(status: tunnelProviderSession.status)
if (tunnelProviderSession.status == .connected && defaults.bool(forKey: kHasSeenInitialFirewallConnectedDialog) == false) {
defaults.set(true, forKey: kHasSeenInitialFirewallConnectedDialog)
self.tapToActivateFirewallLabel.isHidden = true
if (VPNController.shared.status() == .invalid) {
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
@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)
//ensureFirewallWorkingAfterEnabling(waitingSeconds: 5.0)
case .disconnected:
updateFirewallButtonWithStatus(status: .connecting)
FirewallController.shared.setEnabled(true, isUserExplicitToggle: true)
//ensureFirewallWorkingAfterEnabling(waitingSeconds: 5.0)
checkForAskRating()
case .connected:
updateFirewallButtonWithStatus(status: .disconnecting)
FirewallController.shared.setEnabled(false, isUserExplicitToggle: true)
case .connecting, .disconnecting, .reasserting:
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())
}
@IBAction func toggleVPN(_ sender: Any) {
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)
// 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 { (getKey: GetKey) in
try setVPNCredentials(id: getKey.id, keyBase64: getKey.b64)
DDLogInfo("setting VPN creds with ID: \(getKey.id)")
VPNController.shared.setEnabled(true)
}
.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 { (getKey: GetKey) in
try setVPNCredentials(id: getKey.id, keyBase64: getKey.b64)
VPNController.shared.setEnabled(true)
}
.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)
VPNController.shared.setEnabled(true)
}
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: ""))
}
}
}
}
}
@IBAction func viewAuditReportTapped(_ sender: Any) {
showAuditModal()
}
@IBAction func showWhitelist(_ sender: Any) {
performSegue(withIdentifier: "showWhitelist", sender: nil)
}
@IBAction func showSetRegion(_ sender: Any) {
performSegue(withIdentifier: "showSetRegion", sender: nil)
}
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
}
case "showUpgradePlan":
if let vc = segue.destination as? SignupViewController {
if activePlans.isEmpty {
vc.mode = .newSubscription
} else {
vc.mode = .upgrade(active: activePlans)
}
}
default:
break
}
}
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()
}
}
}
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"
}
}
}