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.
		
		
		
		
		
			
		
			
				
	
	
		
			438 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			438 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			Swift
		
	
 | 
						|
// See https://github.com/yapstudios/YapDatabase/wiki/LongLivedReadTransactions and
 | 
						|
// https://github.com/yapstudios/YapDatabase/wiki/YapDatabaseModifiedNotification for
 | 
						|
// more information on database handling.
 | 
						|
 | 
						|
final class HomeVC : BaseVC, UITableViewDataSource, UITableViewDelegate, NewConversationButtonSetDelegate, SeedReminderViewDelegate {
 | 
						|
    private var threads: YapDatabaseViewMappings!
 | 
						|
    private var threadViewModelCache: [String:ThreadViewModel] = [:] // Thread ID to ThreadViewModel
 | 
						|
    private var tableViewTopConstraint: NSLayoutConstraint!
 | 
						|
    
 | 
						|
    private var threadCount: UInt {
 | 
						|
        threads.numberOfItems(inGroup: TSInboxGroup)
 | 
						|
    }
 | 
						|
    
 | 
						|
    private lazy var dbConnection: YapDatabaseConnection = {
 | 
						|
        let result = OWSPrimaryStorage.shared().newDatabaseConnection()
 | 
						|
        result.objectCacheLimit = 500
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    // MARK: UI Components
 | 
						|
    private lazy var seedReminderView: SeedReminderView = {
 | 
						|
        let result = SeedReminderView(hasContinueButton: true)
 | 
						|
        let title = "You're almost finished! 80%"
 | 
						|
        let attributedTitle = NSMutableAttributedString(string: title)
 | 
						|
        attributedTitle.addAttribute(.foregroundColor, value: Colors.accent, range: (title as NSString).range(of: "80%"))
 | 
						|
        result.title = attributedTitle
 | 
						|
        result.subtitle = NSLocalizedString("view_seed_reminder_subtitle_1", comment: "")
 | 
						|
        result.setProgress(0.8, animated: false)
 | 
						|
        result.delegate = self
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
        
 | 
						|
    private lazy var tableView: UITableView = {
 | 
						|
        let result = UITableView()
 | 
						|
        result.backgroundColor = .clear
 | 
						|
        result.separatorStyle = .none
 | 
						|
        result.register(ConversationCell.self, forCellReuseIdentifier: ConversationCell.reuseIdentifier)
 | 
						|
        let bottomInset = Values.newConversationButtonBottomOffset + NewConversationButtonSet.expandedButtonSize + Values.largeSpacing + NewConversationButtonSet.collapsedButtonSize
 | 
						|
        result.contentInset = UIEdgeInsets(top: 0, left: 0, bottom: bottomInset, right: 0)
 | 
						|
        result.showsVerticalScrollIndicator = false
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
 | 
						|
    private lazy var newConversationButtonSet: NewConversationButtonSet = {
 | 
						|
        let result = NewConversationButtonSet()
 | 
						|
        result.delegate = self
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    private lazy var fadeView: UIView = {
 | 
						|
        let result = UIView()
 | 
						|
        let gradient = Gradients.homeVCFade
 | 
						|
        result.setGradient(gradient)
 | 
						|
        result.isUserInteractionEnabled = false
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
 | 
						|
    private lazy var emptyStateView: UIView = {
 | 
						|
        let explanationLabel = UILabel()
 | 
						|
        explanationLabel.textColor = Colors.text
 | 
						|
        explanationLabel.font = .systemFont(ofSize: Values.smallFontSize)
 | 
						|
        explanationLabel.numberOfLines = 0
 | 
						|
        explanationLabel.lineBreakMode = .byWordWrapping
 | 
						|
        explanationLabel.textAlignment = .center
 | 
						|
        explanationLabel.text = NSLocalizedString("vc_home_empty_state_message", comment: "")
 | 
						|
        let createNewPrivateChatButton = Button(style: .prominentOutline, size: .large)
 | 
						|
        createNewPrivateChatButton.setTitle(NSLocalizedString("vc_home_empty_state_button_title", comment: ""), for: UIControl.State.normal)
 | 
						|
        createNewPrivateChatButton.addTarget(self, action: #selector(createNewDM), for: UIControl.Event.touchUpInside)
 | 
						|
        createNewPrivateChatButton.set(.width, to: 196)
 | 
						|
        let result = UIStackView(arrangedSubviews: [ explanationLabel, createNewPrivateChatButton ])
 | 
						|
        result.axis = .vertical
 | 
						|
        result.spacing = Values.mediumSpacing
 | 
						|
        result.alignment = .center
 | 
						|
        result.isHidden = true
 | 
						|
        return result
 | 
						|
    }()
 | 
						|
    
 | 
						|
    // MARK: Lifecycle
 | 
						|
    override func viewDidLoad() {
 | 
						|
        super.viewDidLoad()
 | 
						|
        // Threads (part 1)
 | 
						|
        dbConnection.beginLongLivedReadTransaction() // Freeze the connection for use on the main thread (this gives us a stable data source that doesn't change until we tell it to)
 | 
						|
        // Preparation
 | 
						|
        SignalApp.shared().homeViewController = self
 | 
						|
        // Gradient & nav bar
 | 
						|
        setUpGradientBackground()
 | 
						|
        if navigationController?.navigationBar != nil {
 | 
						|
            setUpNavBarStyle()
 | 
						|
        }
 | 
						|
        updateNavBarButtons()
 | 
						|
        setNavBarTitle("Messages")
 | 
						|
        // Recovery phrase reminder
 | 
						|
        let hasViewedSeed = UserDefaults.standard[.hasViewedSeed]
 | 
						|
        if !hasViewedSeed {
 | 
						|
            view.addSubview(seedReminderView)
 | 
						|
            seedReminderView.pin(.leading, to: .leading, of: view)
 | 
						|
            seedReminderView.pin(.top, to: .top, of: view)
 | 
						|
            seedReminderView.pin(.trailing, to: .trailing, of: view)
 | 
						|
        }
 | 
						|
        // Table view
 | 
						|
        tableView.dataSource = self
 | 
						|
        tableView.delegate = self
 | 
						|
        view.addSubview(tableView)
 | 
						|
        tableView.pin(.leading, to: .leading, of: view)
 | 
						|
        if !hasViewedSeed {
 | 
						|
            tableViewTopConstraint = tableView.pin(.top, to: .bottom, of: seedReminderView)
 | 
						|
        } else {
 | 
						|
            tableViewTopConstraint = tableView.pin(.top, to: .top, of: view, withInset: Values.smallSpacing)
 | 
						|
        }
 | 
						|
        tableView.pin(.trailing, to: .trailing, of: view)
 | 
						|
        tableView.pin(.bottom, to: .bottom, of: view)
 | 
						|
        view.addSubview(fadeView)
 | 
						|
        fadeView.pin(.leading, to: .leading, of: view)
 | 
						|
        let topInset = 0.15 * view.height()
 | 
						|
        fadeView.pin(.top, to: .top, of: view, withInset: topInset)
 | 
						|
        fadeView.pin(.trailing, to: .trailing, of: view)
 | 
						|
        fadeView.pin(.bottom, to: .bottom, of: view)
 | 
						|
        // Empty state view
 | 
						|
        view.addSubview(emptyStateView)
 | 
						|
        emptyStateView.center(.horizontal, in: view)
 | 
						|
        let verticalCenteringConstraint = emptyStateView.center(.vertical, in: view)
 | 
						|
        verticalCenteringConstraint.constant = -16 // Makes things appear centered visually
 | 
						|
        // New conversation button set
 | 
						|
        view.addSubview(newConversationButtonSet)
 | 
						|
        newConversationButtonSet.center(.horizontal, in: view)
 | 
						|
        newConversationButtonSet.pin(.bottom, to: .bottom, of: view, withInset: -Values.newConversationButtonBottomOffset) // Negative due to how the constraint is set up
 | 
						|
        // Notifications
 | 
						|
        let notificationCenter = NotificationCenter.default
 | 
						|
        notificationCenter.addObserver(self, selector: #selector(handleYapDatabaseModifiedNotification(_:)), name: .YapDatabaseModified, object: OWSPrimaryStorage.shared().dbNotificationObject)
 | 
						|
        notificationCenter.addObserver(self, selector: #selector(handleProfileDidChangeNotification(_:)), name: NSNotification.Name(rawValue: kNSNotificationName_OtherUsersProfileDidChange), object: nil)
 | 
						|
        notificationCenter.addObserver(self, selector: #selector(handleLocalProfileDidChangeNotification(_:)), name: Notification.Name(kNSNotificationName_LocalProfileDidChange), object: nil)
 | 
						|
        notificationCenter.addObserver(self, selector: #selector(handleSeedViewedNotification(_:)), name: .seedViewed, object: nil)
 | 
						|
        notificationCenter.addObserver(self, selector: #selector(handleBlockedContactsUpdatedNotification(_:)), name: .blockedContactsUpdated, object: nil)
 | 
						|
        // Threads (part 2)
 | 
						|
        threads = YapDatabaseViewMappings(groups: [ TSInboxGroup ], view: TSThreadDatabaseViewExtensionName) // The extension should be registered at this point
 | 
						|
        threads.setIsReversed(true, forGroup: TSInboxGroup)
 | 
						|
        dbConnection.read { transaction in
 | 
						|
            self.threads.update(with: transaction) // Perform the initial update
 | 
						|
        }
 | 
						|
        // Start polling if needed (i.e. if the user just created or restored their Session ID)
 | 
						|
        if OWSIdentityManager.shared().identityKeyPair() != nil {
 | 
						|
            let appDelegate = UIApplication.shared.delegate as! AppDelegate
 | 
						|
            appDelegate.startPollerIfNeeded()
 | 
						|
            appDelegate.startClosedGroupPollerIfNeeded()
 | 
						|
            appDelegate.startOpenGroupPollersIfNeeded()
 | 
						|
            // Do this only if we created a new Session ID, or if we already received the initial configuration message
 | 
						|
            if UserDefaults.standard[.hasSyncedInitialConfiguration] {
 | 
						|
                appDelegate.syncConfigurationIfNeeded()
 | 
						|
            }
 | 
						|
        }
 | 
						|
        // Onion request path countries cache
 | 
						|
        DispatchQueue.global(qos: .utility).async {
 | 
						|
            let _ = IP2Country.shared.populateCacheIfNeeded()
 | 
						|
        }
 | 
						|
        // Get default open group rooms if needed
 | 
						|
        if Features.useV2OpenGroups {
 | 
						|
            OpenGroupAPIV2.getDefaultRoomsIfNeeded()
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    override func viewDidAppear(_ animated: Bool) {
 | 
						|
        super.viewDidAppear(animated)
 | 
						|
        reload()
 | 
						|
    }
 | 
						|
    
 | 
						|
    deinit {
 | 
						|
        NotificationCenter.default.removeObserver(self)
 | 
						|
    }
 | 
						|
    
 | 
						|
    // MARK: Table View Data Source
 | 
						|
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
 | 
						|
        return Int(threadCount)
 | 
						|
    }
 | 
						|
    
 | 
						|
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
 | 
						|
        let cell = tableView.dequeueReusableCell(withIdentifier: ConversationCell.reuseIdentifier) as! ConversationCell
 | 
						|
        cell.threadViewModel = threadViewModel(at: indexPath.row)
 | 
						|
        return cell
 | 
						|
    }
 | 
						|
        
 | 
						|
    // MARK: Updating
 | 
						|
    private func reload() {
 | 
						|
        AssertIsOnMainThread()
 | 
						|
        dbConnection.beginLongLivedReadTransaction() // Jump to the latest commit
 | 
						|
        dbConnection.read { transaction in
 | 
						|
            self.threads.update(with: transaction)
 | 
						|
        }
 | 
						|
        threadViewModelCache.removeAll()
 | 
						|
        tableView.reloadData()
 | 
						|
        emptyStateView.isHidden = (threadCount != 0)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func handleYapDatabaseModifiedNotification(_ yapDatabase: YapDatabase) {
 | 
						|
        // This code is very finicky and crashes easily
 | 
						|
        AssertIsOnMainThread()
 | 
						|
        let notifications = dbConnection.beginLongLivedReadTransaction() // Jump to the latest commit
 | 
						|
        guard !notifications.isEmpty else { return }
 | 
						|
        let ext = dbConnection.ext(TSThreadDatabaseViewExtensionName) as! YapDatabaseViewConnection
 | 
						|
        let hasChanges = ext.hasChanges(forGroup: TSInboxGroup, in: notifications)
 | 
						|
        guard hasChanges else { return }
 | 
						|
        if let firstChangeSet = notifications[0].userInfo {
 | 
						|
            let firstSnapshot = firstChangeSet[YapDatabaseSnapshotKey] as! UInt64
 | 
						|
            if threads.snapshotOfLastUpdate != firstSnapshot - 1 {
 | 
						|
                return reload() // The code below will crash if we try to process multiple commits at once
 | 
						|
            }
 | 
						|
        }
 | 
						|
        var sectionChanges = NSArray()
 | 
						|
        var rowChanges = NSArray()
 | 
						|
        ext.getSectionChanges(§ionChanges, rowChanges: &rowChanges, for: notifications, with: threads)
 | 
						|
        guard sectionChanges.count > 0 || rowChanges.count > 0 else { return }
 | 
						|
        tableView.beginUpdates()
 | 
						|
        rowChanges.forEach { rowChange in
 | 
						|
            let rowChange = rowChange as! YapDatabaseViewRowChange
 | 
						|
            let key = rowChange.collectionKey.key
 | 
						|
            threadViewModelCache[key] = nil
 | 
						|
            switch rowChange.type {
 | 
						|
            case .delete: tableView.deleteRows(at: [ rowChange.indexPath! ], with: UITableView.RowAnimation.automatic)
 | 
						|
            case .insert: tableView.insertRows(at: [ rowChange.newIndexPath! ], with: UITableView.RowAnimation.automatic)
 | 
						|
            case .move: tableView.moveRow(at: rowChange.indexPath!, to: rowChange.newIndexPath!)
 | 
						|
            case .update: tableView.reloadRows(at: [ rowChange.indexPath! ], with: UITableView.RowAnimation.automatic)
 | 
						|
            default: break
 | 
						|
            }
 | 
						|
        }
 | 
						|
        tableView.endUpdates()
 | 
						|
        emptyStateView.isHidden = (threadCount != 0)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func handleProfileDidChangeNotification(_ notification: Notification) {
 | 
						|
        tableView.reloadData() // TODO: Just reload the affected cell
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func handleLocalProfileDidChangeNotification(_ notification: Notification) {
 | 
						|
        updateNavBarButtons()
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func handleSeedViewedNotification(_ notification: Notification) {
 | 
						|
        tableViewTopConstraint.isActive = false
 | 
						|
        tableViewTopConstraint = tableView.pin(.top, to: .top, of: view, withInset: Values.smallSpacing)
 | 
						|
        seedReminderView.removeFromSuperview()
 | 
						|
    }
 | 
						|
 | 
						|
    @objc private func handleBlockedContactsUpdatedNotification(_ notification: Notification) {
 | 
						|
        self.tableView.reloadData() // TODO: Just reload the affected cell
 | 
						|
    }
 | 
						|
    
 | 
						|
    private func updateNavBarButtons() {
 | 
						|
        let profilePictureSize = Values.verySmallProfilePictureSize
 | 
						|
        let profilePictureView = ProfilePictureView()
 | 
						|
        profilePictureView.accessibilityLabel = "Settings button"
 | 
						|
        profilePictureView.size = profilePictureSize
 | 
						|
        profilePictureView.publicKey = getUserHexEncodedPublicKey()
 | 
						|
        profilePictureView.update()
 | 
						|
        profilePictureView.set(.width, to: profilePictureSize)
 | 
						|
        profilePictureView.set(.height, to: profilePictureSize)
 | 
						|
        let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(openSettings))
 | 
						|
        profilePictureView.addGestureRecognizer(tapGestureRecognizer)
 | 
						|
        let profilePictureViewContainer = UIView()
 | 
						|
        profilePictureViewContainer.accessibilityLabel = "Settings button"
 | 
						|
        profilePictureViewContainer.addSubview(profilePictureView)
 | 
						|
        profilePictureView.pin(.leading, to: .leading, of: profilePictureViewContainer, withInset: 4)
 | 
						|
        profilePictureView.pin(.top, to: .top, of: profilePictureViewContainer)
 | 
						|
        profilePictureView.pin(.trailing, to: .trailing, of: profilePictureViewContainer)
 | 
						|
        profilePictureView.pin(.bottom, to: .bottom, of: profilePictureViewContainer)
 | 
						|
        let leftBarButtonItem = UIBarButtonItem(customView: profilePictureViewContainer)
 | 
						|
        leftBarButtonItem.accessibilityLabel = "Settings button"
 | 
						|
        leftBarButtonItem.isAccessibilityElement = true
 | 
						|
        navigationItem.leftBarButtonItem = leftBarButtonItem
 | 
						|
        let pathStatusViewContainer = UIView()
 | 
						|
        pathStatusViewContainer.accessibilityLabel = "Current onion routing path button"
 | 
						|
        let pathStatusViewContainerSize = Values.verySmallProfilePictureSize // Match the profile picture view
 | 
						|
        pathStatusViewContainer.set(.width, to: pathStatusViewContainerSize)
 | 
						|
        pathStatusViewContainer.set(.height, to: pathStatusViewContainerSize)
 | 
						|
        let pathStatusView = PathStatusView()
 | 
						|
        pathStatusView.accessibilityLabel = "Current onion routing path button"
 | 
						|
        pathStatusView.set(.width, to: PathStatusView.size)
 | 
						|
        pathStatusView.set(.height, to: PathStatusView.size)
 | 
						|
        pathStatusViewContainer.addSubview(pathStatusView)
 | 
						|
        pathStatusView.center(.horizontal, in: pathStatusViewContainer)
 | 
						|
        pathStatusView.center(.vertical, in: pathStatusViewContainer)
 | 
						|
        pathStatusViewContainer.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(showPath)))
 | 
						|
        let rightBarButtonItem = UIBarButtonItem(customView: pathStatusViewContainer)
 | 
						|
        rightBarButtonItem.accessibilityLabel = "Current onion routing path button"
 | 
						|
        rightBarButtonItem.isAccessibilityElement  = true
 | 
						|
        navigationItem.rightBarButtonItem = rightBarButtonItem
 | 
						|
    }
 | 
						|
 | 
						|
    @objc override internal func handleAppModeChangedNotification(_ notification: Notification) {
 | 
						|
        super.handleAppModeChangedNotification(notification)
 | 
						|
        let gradient = Gradients.homeVCFade
 | 
						|
        fadeView.setGradient(gradient) // Re-do the gradient
 | 
						|
        tableView.reloadData()
 | 
						|
    }
 | 
						|
    
 | 
						|
    // MARK: Interaction
 | 
						|
    func handleContinueButtonTapped(from seedReminderView: SeedReminderView) {
 | 
						|
        let seedVC = SeedVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: seedVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
 | 
						|
        guard let thread = self.thread(at: indexPath.row) else { return }
 | 
						|
        show(thread, with: ConversationViewAction.none, highlightedMessageID: nil, animated: true)
 | 
						|
        tableView.deselectRow(at: indexPath, animated: true)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc func show(_ thread: TSThread, with action: ConversationViewAction, highlightedMessageID: String?, animated: Bool) {
 | 
						|
        DispatchMainThreadSafe {
 | 
						|
            if let presentedVC = self.presentedViewController {
 | 
						|
                presentedVC.dismiss(animated: false, completion: nil)
 | 
						|
            }
 | 
						|
            let conversationVC = ConversationVC(thread: thread)
 | 
						|
            self.navigationController?.setViewControllers([ self, conversationVC ], animated: true)
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool {
 | 
						|
        return true
 | 
						|
    }
 | 
						|
    
 | 
						|
    func tableView(_ tableView: UITableView, editActionsForRowAt indexPath: IndexPath) -> [UITableViewRowAction]? {
 | 
						|
        guard let thread = self.thread(at: indexPath.row) else { return [] }
 | 
						|
        let delete = UITableViewRowAction(style: .destructive, title: NSLocalizedString("TXT_DELETE_TITLE", comment: "")) { [weak self] _, _ in
 | 
						|
            var message = NSLocalizedString("CONVERSATION_DELETE_CONFIRMATION_ALERT_MESSAGE", comment: "")
 | 
						|
            if let thread = thread as? TSGroupThread, thread.isClosedGroup, thread.groupModel.groupAdminIds.contains(getUserHexEncodedPublicKey()) {
 | 
						|
                message = "Because you are the creator of this group it will be deleted for everyone. This cannot be undone."
 | 
						|
            }
 | 
						|
            let alert = UIAlertController(title: NSLocalizedString("CONVERSATION_DELETE_CONFIRMATION_ALERT_TITLE", comment: ""), message: message, preferredStyle: .alert)
 | 
						|
            alert.addAction(UIAlertAction(title: NSLocalizedString("TXT_DELETE_TITLE", comment: ""), style: .destructive) { [weak self] _ in
 | 
						|
                self?.delete(thread)
 | 
						|
            })
 | 
						|
            alert.addAction(UIAlertAction(title: NSLocalizedString("TXT_CANCEL_TITLE", comment: ""), style: .default) { _ in })
 | 
						|
            guard let self = self else { return }
 | 
						|
            self.present(alert, animated: true, completion: nil)
 | 
						|
        }
 | 
						|
        delete.backgroundColor = Colors.destructive
 | 
						|
        if thread is TSContactThread {
 | 
						|
            let publicKey = thread.contactIdentifier()!
 | 
						|
            let blockingManager = SSKEnvironment.shared.blockingManager
 | 
						|
            let isBlocked = blockingManager.isRecipientIdBlocked(publicKey)
 | 
						|
            let block = UITableViewRowAction(style: .normal, title: NSLocalizedString("BLOCK_LIST_BLOCK_BUTTON", comment: "")) { _, _ in
 | 
						|
                blockingManager.addBlockedPhoneNumber(publicKey)
 | 
						|
                tableView.reloadRows(at: [ indexPath ], with: UITableView.RowAnimation.fade)
 | 
						|
            }
 | 
						|
            block.backgroundColor = Colors.unimportant
 | 
						|
            let unblock = UITableViewRowAction(style: .normal, title: NSLocalizedString("BLOCK_LIST_UNBLOCK_BUTTON", comment: "")) { _, _ in
 | 
						|
                blockingManager.removeBlockedPhoneNumber(publicKey)
 | 
						|
                tableView.reloadRows(at: [ indexPath ], with: UITableView.RowAnimation.fade)
 | 
						|
            }
 | 
						|
            unblock.backgroundColor = Colors.unimportant
 | 
						|
            return [ delete, (isBlocked ? unblock : block) ]
 | 
						|
        } else {
 | 
						|
            return [ delete ]
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    private func delete(_ thread: TSThread) {
 | 
						|
        let openGroupV2 = Storage.shared.getV2OpenGroup(for: thread.uniqueId!)
 | 
						|
        let openGroup = Storage.shared.getOpenGroup(for: thread.uniqueId!)
 | 
						|
        Storage.write { transaction in
 | 
						|
            Storage.shared.cancelPendingMessageSendJobs(for: thread.uniqueId!, using: transaction)
 | 
						|
            if let openGroupV2 = openGroupV2 {
 | 
						|
                OpenGroupManagerV2.shared.delete(openGroupV2, associatedWith: thread, using: transaction)
 | 
						|
            } else if let openGroup = openGroup {
 | 
						|
                OpenGroupManager.shared.delete(openGroup, associatedWith: thread, using: transaction)
 | 
						|
            } else if let thread = thread as? TSGroupThread, thread.isClosedGroup == true {
 | 
						|
                let groupID = thread.groupModel.groupId
 | 
						|
                let groupPublicKey = LKGroupUtilities.getDecodedGroupID(groupID)
 | 
						|
                do {
 | 
						|
                    try MessageSender.leave(groupPublicKey, using: transaction)
 | 
						|
                } catch {
 | 
						|
                    // TODO: Handle
 | 
						|
                }
 | 
						|
                thread.removeAllThreadInteractions(with: transaction)
 | 
						|
                thread.remove(with: transaction)
 | 
						|
            } else {
 | 
						|
                thread.removeAllThreadInteractions(with: transaction)
 | 
						|
                thread.remove(with: transaction)
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func openSettings() {
 | 
						|
        let settingsVC = SettingsVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: settingsVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc private func showPath() {
 | 
						|
        let pathVC = PathVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: pathVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc func joinOpenGroup() {
 | 
						|
        let joinOpenGroupVC = JoinOpenGroupVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: joinOpenGroupVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc func createNewDM() {
 | 
						|
        let newDMVC = NewDMVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: newDMVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    @objc func createClosedGroup() {
 | 
						|
        let newClosedGroupVC = NewClosedGroupVC()
 | 
						|
        let navigationController = OWSNavigationController(rootViewController: newClosedGroupVC)
 | 
						|
        present(navigationController, animated: true, completion: nil)
 | 
						|
    }
 | 
						|
    
 | 
						|
    // MARK: Convenience
 | 
						|
    private func thread(at index: Int) -> TSThread? {
 | 
						|
        var thread: TSThread? = nil
 | 
						|
        dbConnection.read { transaction in
 | 
						|
            let ext = transaction.ext(TSThreadDatabaseViewExtensionName) as! YapDatabaseViewTransaction
 | 
						|
            thread = ext.object(atRow: UInt(index), inSection: 0, with: self.threads) as! TSThread?
 | 
						|
        }
 | 
						|
        return thread
 | 
						|
    }
 | 
						|
    
 | 
						|
    private func threadViewModel(at index: Int) -> ThreadViewModel? {
 | 
						|
        guard let thread = thread(at: index) else { return nil }
 | 
						|
        if let cachedThreadViewModel = threadViewModelCache[thread.uniqueId!] {
 | 
						|
            return cachedThreadViewModel
 | 
						|
        } else {
 | 
						|
            var threadViewModel: ThreadViewModel? = nil
 | 
						|
            dbConnection.read { transaction in
 | 
						|
                threadViewModel = ThreadViewModel(thread: thread, transaction: transaction)
 | 
						|
            }
 | 
						|
            threadViewModelCache[thread.uniqueId!] = threadViewModel
 | 
						|
            return threadViewModel
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |