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.
		
		
		
		
		
			
		
			
				
	
	
		
			168 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			168 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Swift
		
	
| import SessionUIKit
 | |
| 
 | |
| final class ThreadPickerVC : UIViewController, UITableViewDataSource, UITableViewDelegate, AttachmentApprovalViewControllerDelegate {
 | |
|     private var threads: YapDatabaseViewMappings!
 | |
|     private var threadViewModelCache: [String:ThreadViewModel] = [:] // Thread ID to ThreadViewModel
 | |
|     private var selectedThread: TSThread?
 | |
|     var shareVC: ShareVC?
 | |
|     
 | |
|     private var threadCount: UInt {
 | |
|         threads.numberOfItems(inGroup: TSInboxGroup)
 | |
|     }
 | |
|     
 | |
|     private lazy var dbConnection: YapDatabaseConnection = {
 | |
|         let result = OWSPrimaryStorage.shared().newDatabaseConnection()
 | |
|         result.objectCacheLimit = 500
 | |
|         return result
 | |
|     }()
 | |
| 
 | |
|     private lazy var tableView: UITableView = {
 | |
|         let result = UITableView()
 | |
|         result.backgroundColor = .clear
 | |
|         result.separatorStyle = .none
 | |
|         result.register(SimplifiedConversationCell.self, forCellReuseIdentifier: SimplifiedConversationCell.reuseIdentifier)
 | |
|         result.showsVerticalScrollIndicator = false
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     private lazy var fadeView: UIView = {
 | |
|         let result = UIView()
 | |
|         let gradient = Gradients.homeVCFade
 | |
|         result.setGradient(gradient)
 | |
|         result.isUserInteractionEnabled = false
 | |
|         return result
 | |
|     }()
 | |
|     
 | |
|     // MARK: Lifecycle
 | |
|     override func viewDidLoad() {
 | |
|         super.viewDidLoad()
 | |
|         setupNavBar()
 | |
|         // Gradient
 | |
|         view.backgroundColor = .clear
 | |
|         let gradient = Gradients.defaultBackground
 | |
|         view.setGradient(gradient)
 | |
|         // Threads
 | |
|         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)
 | |
|         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
 | |
|         }
 | |
|         // Title
 | |
|         let titleLabel = UILabel()
 | |
|         titleLabel.text = NSLocalizedString("vc_share_title", comment: "")
 | |
|         titleLabel.textColor = Colors.text
 | |
|         titleLabel.font = .boldSystemFont(ofSize: Values.veryLargeFontSize)
 | |
|         navigationItem.titleView = titleLabel
 | |
|         // Table view
 | |
|         tableView.dataSource = self
 | |
|         tableView.delegate = self
 | |
|         view.addSubview(tableView)
 | |
|         tableView.pin(to: 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)
 | |
|         // Reload
 | |
|         reload()
 | |
|     }
 | |
|     
 | |
|     private func setupNavBar() {
 | |
|         guard let navigationBar = navigationController?.navigationBar else { return }
 | |
|         if #available(iOS 15.0, *) {
 | |
|             let appearance = UINavigationBarAppearance()
 | |
|             appearance.configureWithOpaqueBackground()
 | |
|             appearance.backgroundColor = Colors.navigationBarBackground
 | |
|             navigationBar.standardAppearance = appearance;
 | |
|             navigationBar.scrollEdgeAppearance = navigationBar.standardAppearance
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // 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: SimplifiedConversationCell.reuseIdentifier) as! SimplifiedConversationCell
 | |
|         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()
 | |
|     }
 | |
|     
 | |
|     // MARK: Interaction
 | |
|     func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
 | |
|         guard let thread = self.thread(at: indexPath.row), let attachments = ShareVC.attachmentPrepPromise?.value else { return }
 | |
|         self.selectedThread = thread
 | |
|         tableView.deselectRow(at: indexPath, animated: true)
 | |
|         let approvalVC = AttachmentApprovalViewController.wrappedInNavController(attachments: attachments, approvalDelegate: self)
 | |
|         navigationController!.present(approvalVC, animated: true, completion: nil)
 | |
|     }
 | |
|     
 | |
|     func attachmentApproval(_ attachmentApproval: AttachmentApprovalViewController, didApproveAttachments attachments: [SignalAttachment], messageText: String?) {
 | |
|         let message = VisibleMessage()
 | |
|         message.sentTimestamp = NSDate.millisecondTimestamp()
 | |
|         message.text = messageText
 | |
|         let tsMessage = TSOutgoingMessage.from(message, associatedWith: selectedThread!)
 | |
|         Storage.write { transaction in
 | |
|             tsMessage.save(with: transaction)
 | |
|         }
 | |
|         shareVC!.dismiss(animated: true, completion: nil)
 | |
|         ModalActivityIndicatorViewController.present(fromViewController: shareVC!, canCancel: false, message: NSLocalizedString("vc_share_sending_message", comment: "")) { activityIndicator in
 | |
|             MessageSender.sendNonDurably(message, with: attachments, in: self.selectedThread!).done { [weak self] _ in
 | |
|                 guard let self = self else { return }
 | |
|                 activityIndicator.dismiss { }
 | |
|                 self.shareVC!.shareViewWasCompleted()
 | |
|             }.catch { [weak self] error in
 | |
|                 guard let self = self else { return }
 | |
|                 activityIndicator.dismiss { }
 | |
|                 self.shareVC!.shareViewFailed(error: error)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     func attachmentApprovalDidCancel(_ attachmentApproval: AttachmentApprovalViewController) {
 | |
|         // Do nothing
 | |
|     }
 | |
| 
 | |
|     func attachmentApproval(_ attachmentApproval: AttachmentApprovalViewController, didChangeMessageText newMessageText: String?) {
 | |
|         // Do nothing
 | |
|     }
 | |
|     
 | |
|     // 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
 | |
|         }
 | |
|     }
 | |
| }
 |