mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			240 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			240 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Swift
		
	
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | 
						|
 | 
						|
import UIKit
 | 
						|
import SessionUIKit
 | 
						|
import SessionSnodeKit
 | 
						|
import SessionMessagingKit
 | 
						|
import SignalUtilitiesKit
 | 
						|
 | 
						|
@objc(LKNukeDataModal)
 | 
						|
final class NukeDataModal: Modal {
 | 
						|
    
 | 
						|
    // MARK: - Components
 | 
						|
    
 | 
						|
    private lazy var titleLabel: UILabel = {
 | 
						|
        let result = UILabel()
 | 
						|
        result.textColor = Colors.text
 | 
						|
        result.font = .boldSystemFont(ofSize: Values.mediumFontSize)
 | 
						|
        result.text = "modal_clear_all_data_title".localized()
 | 
						|
        result.numberOfLines = 0
 | 
						|
        result.lineBreakMode = .byWordWrapping
 | 
						|
        result.textAlignment = .center
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var explanationLabel: UILabel = {
 | 
						|
        let result = UILabel()
 | 
						|
        result.textColor = Colors.text.withAlphaComponent(Values.mediumOpacity)
 | 
						|
        result.font = .systemFont(ofSize: Values.smallFontSize)
 | 
						|
        result.text = "modal_clear_all_data_explanation".localized()
 | 
						|
        result.numberOfLines = 0
 | 
						|
        result.textAlignment = .center
 | 
						|
        result.lineBreakMode = .byWordWrapping
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var clearDataButton: UIButton = {
 | 
						|
        let result = UIButton()
 | 
						|
        result.set(.height, to: Values.mediumButtonHeight)
 | 
						|
        result.layer.cornerRadius = Modal.buttonCornerRadius
 | 
						|
        if isDarkMode {
 | 
						|
            result.backgroundColor = Colors.destructive
 | 
						|
        }
 | 
						|
        result.titleLabel!.font = .systemFont(ofSize: Values.smallFontSize)
 | 
						|
        result.setTitleColor(isLightMode ? Colors.destructive : Colors.text, for: UIControl.State.normal)
 | 
						|
        result.setTitle("TXT_DELETE_TITLE".localized(), for: UIControl.State.normal)
 | 
						|
        result.addTarget(self, action: #selector(clearAllData), for: UIControl.Event.touchUpInside)
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var buttonStackView1: UIStackView = {
 | 
						|
        let result = UIStackView(arrangedSubviews: [ cancelButton, clearDataButton ])
 | 
						|
        result.axis = .horizontal
 | 
						|
        result.spacing = Values.mediumSpacing
 | 
						|
        result.distribution = .fillEqually
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var deviceOnlyButton: UIButton = {
 | 
						|
        let result = UIButton()
 | 
						|
        result.set(.height, to: Values.mediumButtonHeight)
 | 
						|
        result.layer.cornerRadius = Modal.buttonCornerRadius
 | 
						|
        result.backgroundColor = Colors.buttonBackground
 | 
						|
        result.titleLabel!.font = .systemFont(ofSize: Values.smallFontSize)
 | 
						|
        result.setTitleColor(Colors.text, for: UIControl.State.normal)
 | 
						|
        result.setTitle("modal_clear_all_data_device_only_button_title".localized(), for: UIControl.State.normal)
 | 
						|
        result.addTarget(self, action: #selector(clearDeviceOnly), for: UIControl.Event.touchUpInside)
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var entireAccountButton: UIButton = {
 | 
						|
        let result = UIButton()
 | 
						|
        result.set(.height, to: Values.mediumButtonHeight)
 | 
						|
        result.layer.cornerRadius = Modal.buttonCornerRadius
 | 
						|
        if isDarkMode {
 | 
						|
            result.backgroundColor = Colors.destructive
 | 
						|
        }
 | 
						|
        result.titleLabel!.font = .systemFont(ofSize: Values.smallFontSize)
 | 
						|
        result.setTitleColor(isLightMode ? Colors.destructive : Colors.text, for: UIControl.State.normal)
 | 
						|
        result.setTitle("modal_clear_all_data_entire_account_button_title".localized(), for: UIControl.State.normal)
 | 
						|
        result.addTarget(self, action: #selector(clearEntireAccount), for: UIControl.Event.touchUpInside)
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var buttonStackView2: UIStackView = {
 | 
						|
        let result = UIStackView(arrangedSubviews: [ deviceOnlyButton, entireAccountButton ])
 | 
						|
        result.axis = .horizontal
 | 
						|
        result.spacing = Values.mediumSpacing
 | 
						|
        result.distribution = .fillEqually
 | 
						|
        result.alpha = 0
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var buttonStackViewContainer: UIView = {
 | 
						|
        let result = UIView()
 | 
						|
        result.addSubview(buttonStackView2)
 | 
						|
        buttonStackView2.pin(to: result)
 | 
						|
        result.addSubview(buttonStackView1)
 | 
						|
        buttonStackView1.pin(to: result)
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var contentStackView: UIStackView = {
 | 
						|
        let result = UIStackView(arrangedSubviews: [ titleLabel, explanationLabel ])
 | 
						|
        result.axis = .vertical
 | 
						|
        result.spacing = Values.largeSpacing
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var mainStackView: UIStackView = {
 | 
						|
        let result = UIStackView(arrangedSubviews: [ contentStackView, buttonStackViewContainer ])
 | 
						|
        result.axis = .vertical
 | 
						|
        result.spacing = Values.largeSpacing - Values.smallFontSize / 2
 | 
						|
        
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    // MARK: - Lifecycle
 | 
						|
    
 | 
						|
    override func populateContentView() {
 | 
						|
        contentView.addSubview(mainStackView)
 | 
						|
        mainStackView.pin(.leading, to: .leading, of: contentView, withInset: Values.largeSpacing)
 | 
						|
        mainStackView.pin(.top, to: .top, of: contentView, withInset: Values.largeSpacing)
 | 
						|
        contentView.pin(.trailing, to: .trailing, of: mainStackView, withInset: Values.largeSpacing)
 | 
						|
        contentView.pin(.bottom, to: .bottom, of: mainStackView, withInset: mainStackView.spacing)
 | 
						|
    }
 | 
						|
    
 | 
						|
    // MARK: - Interaction
 | 
						|
    
 | 
						|
    @objc private func clearAllData() {
 | 
						|
        UIView.animate(withDuration: 0.25) {
 | 
						|
            self.buttonStackView1.alpha = 0
 | 
						|
            self.buttonStackView2.alpha = 1
 | 
						|
        }
 | 
						|
        
 | 
						|
        UIView.transition(
 | 
						|
            with: explanationLabel,
 | 
						|
            duration: 0.25,
 | 
						|
            options: .transitionCrossDissolve,
 | 
						|
            animations: {
 | 
						|
                self.explanationLabel.text = "modal_clear_all_data_explanation_2".localized()
 | 
						|
            },
 | 
						|
            completion: nil
 | 
						|
        )
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func clearDeviceOnly() {
 | 
						|
        ModalActivityIndicatorViewController.present(fromViewController: self, canCancel: false) { [weak self] _ in
 | 
						|
            Storage.shared
 | 
						|
                .writeAsync { db in try MessageSender.syncConfiguration(db, forceSyncNow: true) }
 | 
						|
                .ensure(on: DispatchQueue.main) {
 | 
						|
                    self?.deleteAllLocalData()
 | 
						|
                    self?.dismiss(animated: true, completion: nil) // Dismiss the loader
 | 
						|
                }
 | 
						|
                .retainUntilComplete()
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func clearEntireAccount() {
 | 
						|
        ModalActivityIndicatorViewController
 | 
						|
            .present(fromViewController: self, canCancel: false) { [weak self] _ in
 | 
						|
                SnodeAPI.clearAllData()
 | 
						|
                    .done(on: DispatchQueue.main) { confirmations in
 | 
						|
                        self?.dismiss(animated: true, completion: nil) // Dismiss the loader
 | 
						|
                        
 | 
						|
                        let potentiallyMaliciousSnodes = confirmations.compactMap { $0.value == false ? $0.key : nil }
 | 
						|
                        
 | 
						|
                        if potentiallyMaliciousSnodes.isEmpty {
 | 
						|
                            self?.deleteAllLocalData()
 | 
						|
                        }
 | 
						|
                        else {
 | 
						|
                            let message: String
 | 
						|
                            if potentiallyMaliciousSnodes.count == 1 {
 | 
						|
                                message = String(format: "dialog_clear_all_data_deletion_failed_1".localized(), potentiallyMaliciousSnodes[0])
 | 
						|
                            }
 | 
						|
                            else {
 | 
						|
                                message = String(format: "dialog_clear_all_data_deletion_failed_2".localized(), String(potentiallyMaliciousSnodes.count), potentiallyMaliciousSnodes.joined(separator: ", "))
 | 
						|
                            }
 | 
						|
                            
 | 
						|
                            let alert = UIAlertController(title: "Error", message: message, preferredStyle: .alert)
 | 
						|
                            alert.addAction(UIAlertAction(title: "BUTTON_OK".localized(), style: .default, handler: nil))
 | 
						|
                            
 | 
						|
                            self?.presentAlert(alert)
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                    .catch(on: DispatchQueue.main) { error in
 | 
						|
                        self?.dismiss(animated: true, completion: nil) // Dismiss the loader
 | 
						|
                        
 | 
						|
                        let alert = UIAlertController(title: "Error", message: error.localizedDescription, preferredStyle: .alert)
 | 
						|
                        alert.addAction(UIAlertAction(title: "BUTTON_OK".localized(), style: .default, handler: nil))
 | 
						|
                        self?.presentAlert(alert)
 | 
						|
                    }
 | 
						|
            }
 | 
						|
    }
 | 
						|
    
 | 
						|
    private func deleteAllLocalData() {
 | 
						|
        // Unregister push notifications if needed
 | 
						|
        let isUsingFullAPNs: Bool = UserDefaults.standard[.isUsingFullAPNs]
 | 
						|
        let maybeDeviceToken: String? = UserDefaults.standard[.deviceToken]
 | 
						|
        
 | 
						|
        if isUsingFullAPNs, let deviceToken: String = maybeDeviceToken {
 | 
						|
            let data: Data = Data(hex: deviceToken)
 | 
						|
            PushNotificationAPI.unregister(data).retainUntilComplete()
 | 
						|
        }
 | 
						|
        
 | 
						|
        // Clear the app badge and notifications
 | 
						|
        AppEnvironment.shared.notificationPresenter.clearAllNotifications()
 | 
						|
        CurrentAppContext().setMainAppBadgeNumber(0)
 | 
						|
        
 | 
						|
        // Clear out the user defaults
 | 
						|
        UserDefaults.removeAll()
 | 
						|
        
 | 
						|
        // Remove the cached key so it gets re-cached on next access
 | 
						|
        General.cache.mutate { $0.encodedPublicKey = nil }
 | 
						|
        
 | 
						|
        // Clear the Snode pool
 | 
						|
        SnodeAPI.clearSnodePool()
 | 
						|
        
 | 
						|
        // Stop any pollers
 | 
						|
        (UIApplication.shared.delegate as? AppDelegate)?.stopPollers()
 | 
						|
        
 | 
						|
        // Call through to the SessionApp's "resetAppData" which will wipe out logs, database and
 | 
						|
        // profile storage
 | 
						|
        let wasUnlinked: Bool = UserDefaults.standard[.wasUnlinked]
 | 
						|
        
 | 
						|
        SessionApp.resetAppData {
 | 
						|
            // Resetting the data clears the old user defaults. We need to restore the unlink default.
 | 
						|
            UserDefaults.standard[.wasUnlinked] = wasUnlinked
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |