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.
		
		
		
		
		
			
		
			
				
	
	
		
			175 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			175 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import GRDB
 | |
| import DifferenceKit
 | |
| import SignalUtilitiesKit
 | |
| 
 | |
| public class MessageRequestsViewModel {
 | |
|     public typealias SectionModel = ArraySection<Section, SessionThreadViewModel>
 | |
|     
 | |
|     // MARK: - Section
 | |
|     
 | |
|     public enum Section: Differentiable {
 | |
|         case threads
 | |
|         case loadMore
 | |
|     }
 | |
|     
 | |
|     // MARK: - Variables
 | |
|     
 | |
|     public static let pageSize: Int = 20
 | |
|     
 | |
|     // MARK: - Initialization
 | |
|     
 | |
|     init() {
 | |
|         self.pagedDataObserver = nil
 | |
|         
 | |
|         // Note: Since this references self we need to finish initializing before setting it, we
 | |
|         // also want to skip the initial query and trigger it async so that the push animation
 | |
|         // doesn't stutter (it should load basically immediately but without this there is a
 | |
|         // distinct stutter)
 | |
|         let userPublicKey: String = getUserHexEncodedPublicKey()
 | |
|         let thread: TypedTableAlias<SessionThread> = TypedTableAlias()
 | |
|         self.pagedDataObserver = PagedDatabaseObserver(
 | |
|             pagedTable: SessionThread.self,
 | |
|             pageSize: MessageRequestsViewModel.pageSize,
 | |
|             idColumn: .id,
 | |
|             observedChanges: [
 | |
|                 PagedData.ObservedChanges(
 | |
|                     table: SessionThread.self,
 | |
|                     columns: [
 | |
|                         .id,
 | |
|                         .shouldBeVisible
 | |
|                     ]
 | |
|                 ),
 | |
|                 PagedData.ObservedChanges(
 | |
|                     table: Interaction.self,
 | |
|                     columns: [
 | |
|                         .body,
 | |
|                         .wasRead
 | |
|                     ],
 | |
|                     joinToPagedType: {
 | |
|                         let interaction: TypedTableAlias<Interaction> = TypedTableAlias()
 | |
|                         
 | |
|                         return SQL("LEFT JOIN \(Interaction.self) ON \(interaction[.threadId]) = \(thread[.id])")
 | |
|                     }()
 | |
|                 ),
 | |
|                 PagedData.ObservedChanges(
 | |
|                     table: Contact.self,
 | |
|                     columns: [.isBlocked],
 | |
|                     joinToPagedType: {
 | |
|                         let contact: TypedTableAlias<Contact> = TypedTableAlias()
 | |
|                         
 | |
|                         return SQL("LEFT JOIN \(Contact.self) ON \(contact[.id]) = \(thread[.id])")
 | |
|                     }()
 | |
|                 ),
 | |
|                 PagedData.ObservedChanges(
 | |
|                     table: Profile.self,
 | |
|                     columns: [.name, .nickname, .profilePictureFileName],
 | |
|                     joinToPagedType: {
 | |
|                         let profile: TypedTableAlias<Profile> = TypedTableAlias()
 | |
|                         
 | |
|                         return SQL("LEFT JOIN \(Profile.self) ON \(profile[.id]) = \(thread[.id])")
 | |
|                     }()
 | |
|                 ),
 | |
|                 PagedData.ObservedChanges(
 | |
|                     table: RecipientState.self,
 | |
|                     columns: [.state],
 | |
|                     joinToPagedType: {
 | |
|                         let interaction: TypedTableAlias<Interaction> = TypedTableAlias()
 | |
|                         let recipientState: TypedTableAlias<RecipientState> = TypedTableAlias()
 | |
|                         
 | |
|                         return """
 | |
|                             LEFT JOIN \(Interaction.self) ON \(interaction[.threadId]) = \(thread[.id])
 | |
|                             LEFT JOIN \(RecipientState.self) ON \(recipientState[.interactionId]) = \(interaction[.id])
 | |
|                         """
 | |
|                     }()
 | |
|                 )
 | |
|             ],
 | |
|             /// **Note:** This `optimisedJoinSQL` value includes the required minimum joins needed for the query
 | |
|             joinSQL: SessionThreadViewModel.optimisedJoinSQL,
 | |
|             filterSQL: SessionThreadViewModel.messageRequestsFilterSQL(userPublicKey: userPublicKey),
 | |
|             groupSQL: SessionThreadViewModel.groupSQL,
 | |
|             orderSQL: SessionThreadViewModel.messageRequetsOrderSQL,
 | |
|             dataQuery: SessionThreadViewModel.baseQuery(
 | |
|                 userPublicKey: userPublicKey,
 | |
|                 filterSQL: SessionThreadViewModel.messageRequestsFilterSQL(userPublicKey: userPublicKey),
 | |
|                 groupSQL: SessionThreadViewModel.groupSQL,
 | |
|                 orderSQL: SessionThreadViewModel.messageRequetsOrderSQL
 | |
|             ),
 | |
|             onChangeUnsorted: { [weak self] updatedData, updatedPageInfo in
 | |
|                 guard let updatedThreadData: [SectionModel] = self?.process(data: updatedData, for: updatedPageInfo) else {
 | |
|                     return
 | |
|                 }
 | |
|                 
 | |
|                 // If we have the 'onThreadChange' callback then trigger it, otherwise just store the changes
 | |
|                 // to be sent to the callback if we ever start observing again (when we have the callback it needs
 | |
|                 // to do the data updating as it's tied to UI updates and can cause crashes if not updated in the
 | |
|                 // correct order)
 | |
|                 guard let onThreadChange: (([SectionModel]) -> ()) = self?.onThreadChange else {
 | |
|                     self?.unobservedThreadDataChanges = updatedThreadData
 | |
|                     return
 | |
|                 }
 | |
| 
 | |
|                 onThreadChange(updatedThreadData)
 | |
|             }
 | |
|         )
 | |
|         
 | |
|         // Run the initial query on a background thread so we don't block the push transition
 | |
|         DispatchQueue.global(qos: .default).async { [weak self] in
 | |
|             // The `.pageBefore` will query from a `0` offset loading the first page
 | |
|             self?.pagedDataObserver?.load(.pageBefore)
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // MARK: - Thread Data
 | |
|     
 | |
|     public private(set) var unobservedThreadDataChanges: [SectionModel]?
 | |
|     public private(set) var threadData: [SectionModel] = []
 | |
|     public private(set) var pagedDataObserver: PagedDatabaseObserver<SessionThread, SessionThreadViewModel>?
 | |
|     
 | |
|     public var onThreadChange: (([SectionModel]) -> ())? {
 | |
|         didSet {
 | |
|             // When starting to observe interaction changes we want to trigger a UI update just in case the
 | |
|             // data was changed while we weren't observing
 | |
|             if let unobservedThreadDataChanges: [SectionModel] = self.unobservedThreadDataChanges {
 | |
|                 onThreadChange?(unobservedThreadDataChanges)
 | |
|                 self.unobservedThreadDataChanges = nil
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private func process(data: [SessionThreadViewModel], for pageInfo: PagedData.PageInfo) -> [SectionModel] {
 | |
|         let groupedOldData: [String: [SessionThreadViewModel]] = (self.threadData
 | |
|             .first(where: { $0.model == .threads })?
 | |
|             .elements)
 | |
|             .defaulting(to: [])
 | |
|             .grouped(by: \.threadId)
 | |
|         
 | |
|         return [
 | |
|             [
 | |
|                 SectionModel(
 | |
|                     section: .threads,
 | |
|                     elements: data
 | |
|                         .sorted { lhs, rhs -> Bool in lhs.lastInteractionDate > rhs.lastInteractionDate }
 | |
|                         .map { viewModel -> SessionThreadViewModel in
 | |
|                             viewModel.populatingCurrentUserBlindedKey(
 | |
|                                 currentUserBlindedPublicKeyForThisThread: groupedOldData[viewModel.threadId]?
 | |
|                                     .first?
 | |
|                                     .currentUserBlindedPublicKey
 | |
|                             )
 | |
|                         }
 | |
|                 )
 | |
|             ],
 | |
|             (!data.isEmpty && (pageInfo.pageOffset + pageInfo.currentCount) < pageInfo.totalCount ?
 | |
|                 [SectionModel(section: .loadMore)] :
 | |
|                 []
 | |
|             )
 | |
|         ].flatMap { $0 }
 | |
|     }
 | |
|     
 | |
|     public func updateThreadData(_ updatedData: [SectionModel]) {
 | |
|         self.threadData = updatedData
 | |
|     }
 | |
| }
 |