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.
		
		
		
		
		
			
		
			
	
	
		
			391 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Swift
		
	
		
		
			
		
	
	
			391 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Swift
		
	
| 
											8 years ago
										 | // | ||
| 
											7 years ago
										 | //  Copyright (c) 2019 Open Whisper Systems. All rights reserved. | ||
| 
											8 years ago
										 | // | ||
|  | 
 | ||
|  | import Foundation | ||
| 
											5 years ago
										 | 
 | ||
| 
											8 years ago
										 | 
 | ||
| 
											7 years ago
										 | public typealias MessageSortKey = UInt64 | ||
|  | public struct ConversationSortKey: Comparable { | ||
|  |     let creationDate: Date | ||
|  |     let lastMessageReceivedAtDate: Date? | ||
|  | 
 | ||
|  |     // MARK: Comparable | ||
|  | 
 | ||
|  |     public static func < (lhs: ConversationSortKey, rhs: ConversationSortKey) -> Bool { | ||
|  |         let lhsDate = lhs.lastMessageReceivedAtDate ?? lhs.creationDate | ||
|  |         let rhsDate = rhs.lastMessageReceivedAtDate ?? rhs.creationDate | ||
|  |         return lhsDate < rhsDate | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | public class ConversationSearchResult<SortKey>: Comparable where SortKey: Comparable { | ||
| 
											7 years ago
										 |     public let thread: ThreadViewModel | ||
| 
											7 years ago
										 | 
 | ||
|  |     public let messageId: String? | ||
| 
											7 years ago
										 |     public let messageDate: Date? | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     public let snippet: String? | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     private let sortKey: SortKey | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     init(thread: ThreadViewModel, sortKey: SortKey, messageId: String? = nil, messageDate: Date? = nil, snippet: String? = nil) { | ||
| 
											7 years ago
										 |         self.thread = thread | ||
| 
											7 years ago
										 |         self.sortKey = sortKey | ||
| 
											7 years ago
										 |         self.messageId = messageId | ||
| 
											7 years ago
										 |         self.messageDate = messageDate | ||
| 
											7 years ago
										 |         self.snippet = snippet | ||
| 
											7 years ago
										 |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     // MARK: Comparable | ||
| 
											7 years ago
										 | 
 | ||
|  |     public static func < (lhs: ConversationSearchResult, rhs: ConversationSearchResult) -> Bool { | ||
|  |         return lhs.sortKey < rhs.sortKey | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Equatable | ||
|  | 
 | ||
|  |     public static func == (lhs: ConversationSearchResult, rhs: ConversationSearchResult) -> Bool { | ||
| 
											7 years ago
										 |         return lhs.thread.threadRecord.uniqueId == rhs.thread.threadRecord.uniqueId && | ||
|  |             lhs.messageId == rhs.messageId | ||
| 
											7 years ago
										 |     } | ||
|  | } | ||
|  | 
 | ||
| 
											7 years ago
										 | public class HomeScreenSearchResultSet: NSObject { | ||
| 
											7 years ago
										 |     public let searchText: String | ||
| 
											7 years ago
										 |     public let conversations: [ConversationSearchResult<ConversationSortKey>] | ||
|  |     public let messages: [ConversationSearchResult<MessageSortKey>] | ||
| 
											7 years ago
										 | 
 | ||
| 
											5 years ago
										 |     public init(searchText: String, conversations: [ConversationSearchResult<ConversationSortKey>], messages: [ConversationSearchResult<MessageSortKey>]) { | ||
| 
											7 years ago
										 |         self.searchText = searchText | ||
| 
											7 years ago
										 |         self.conversations = conversations | ||
|  |         self.messages = messages | ||
|  |     } | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     public class var empty: HomeScreenSearchResultSet { | ||
| 
											5 years ago
										 |         return HomeScreenSearchResultSet(searchText: "", conversations: [], messages: []) | ||
| 
											7 years ago
										 |     } | ||
| 
											7 years ago
										 | 
 | ||
|  |     public var isEmpty: Bool { | ||
| 
											5 years ago
										 |         return conversations.isEmpty && messages.isEmpty | ||
| 
											7 years ago
										 |     } | ||
| 
											7 years ago
										 | } | ||
|  | 
 | ||
| 
											7 years ago
										 | @objc | ||
|  | public class GroupSearchResult: NSObject, Comparable { | ||
|  |     public let thread: ThreadViewModel | ||
|  | 
 | ||
|  |     private let sortKey: ConversationSortKey | ||
|  | 
 | ||
|  |     init(thread: ThreadViewModel, sortKey: ConversationSortKey) { | ||
|  |         self.thread = thread | ||
|  |         self.sortKey = sortKey | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Comparable | ||
|  | 
 | ||
|  |     public static func < (lhs: GroupSearchResult, rhs: GroupSearchResult) -> Bool { | ||
|  |         return lhs.sortKey < rhs.sortKey | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Equatable | ||
|  | 
 | ||
|  |     public static func == (lhs: GroupSearchResult, rhs: GroupSearchResult) -> Bool { | ||
|  |         return lhs.thread.threadRecord.uniqueId == rhs.thread.threadRecord.uniqueId | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | @objc | ||
|  | public class ComposeScreenSearchResultSet: NSObject { | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public let searchText: String | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public let groups: [GroupSearchResult] | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public var groupThreads: [TSGroupThread] { | ||
|  |         return groups.compactMap { $0.thread.threadRecord as? TSGroupThread } | ||
|  |     } | ||
|  | 
 | ||
| 
											5 years ago
										 |     public init(searchText: String, groups: [GroupSearchResult]) { | ||
| 
											7 years ago
										 |         self.searchText = searchText | ||
|  |         self.groups = groups | ||
|  |     } | ||
|  | 
 | ||
|  |     @objc | ||
| 
											5 years ago
										 |     public static let empty = ComposeScreenSearchResultSet(searchText: "", groups: []) | ||
| 
											7 years ago
										 | 
 | ||
|  |     @objc | ||
|  |     public var isEmpty: Bool { | ||
| 
											5 years ago
										 |         return groups.isEmpty | ||
| 
											7 years ago
										 |     } | ||
|  | } | ||
|  | 
 | ||
| 
											8 years ago
										 | @objc | ||
| 
											7 years ago
										 | public class MessageSearchResult: NSObject, Comparable { | ||
|  | 
 | ||
|  |     public let messageId: String | ||
|  |     public let sortId: UInt64 | ||
|  | 
 | ||
|  |     init(messageId: String, sortId: UInt64) { | ||
|  |         self.messageId = messageId | ||
|  |         self.sortId = sortId | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: - Comparable | ||
|  | 
 | ||
|  |     public static func < (lhs: MessageSearchResult, rhs: MessageSearchResult) -> Bool { | ||
|  |         return lhs.sortId < rhs.sortId | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | @objc | ||
|  | public class ConversationScreenSearchResultSet: NSObject { | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public let searchText: String | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public let messages: [MessageSearchResult] | ||
|  | 
 | ||
|  |     @objc | ||
|  |     public lazy var messageSortIds: [UInt64] = { | ||
|  |         return messages.map { $0.sortId } | ||
|  |     }() | ||
|  | 
 | ||
|  |     // MARK: Static members | ||
|  | 
 | ||
|  |     public static let empty: ConversationScreenSearchResultSet = ConversationScreenSearchResultSet(searchText: "", messages: []) | ||
|  | 
 | ||
|  |     // MARK: Init | ||
|  | 
 | ||
|  |     public init(searchText: String, messages: [MessageSearchResult]) { | ||
|  |         self.searchText = searchText | ||
|  |         self.messages = messages | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: - CustomDebugStringConvertible | ||
|  | 
 | ||
|  |     override public var debugDescription: String { | ||
|  |         return "ConversationScreenSearchResultSet(searchText: \(searchText), messages: [\(messages.count) matches])" | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | @objc | ||
|  | public class FullTextSearcher: NSObject { | ||
| 
											8 years ago
										 | 
 | ||
| 
											7 years ago
										 |     // MARK: - Dependencies | ||
|  | 
 | ||
|  |     private var tsAccountManager: TSAccountManager { | ||
|  |         return TSAccountManager.sharedInstance() | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: -  | ||
|  | 
 | ||
| 
											7 years ago
										 |     private let finder: FullTextSearchFinder | ||
| 
											7 years ago
										 | 
 | ||
| 
											8 years ago
										 |     @objc | ||
| 
											7 years ago
										 |     public static let shared: FullTextSearcher = FullTextSearcher() | ||
| 
											8 years ago
										 |     override private init() { | ||
| 
											7 years ago
										 |         finder = FullTextSearchFinder() | ||
| 
											8 years ago
										 |         super.init() | ||
|  |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     @objc | ||
|  |     public func searchForComposeScreen(searchText: String, | ||
| 
											5 years ago
										 |                                        transaction: YapDatabaseReadTransaction) -> ComposeScreenSearchResultSet { | ||
| 
											7 years ago
										 | 
 | ||
|  |         var groups: [GroupSearchResult] = [] | ||
|  | 
 | ||
|  |         self.finder.enumerateObjects(searchText: searchText, transaction: transaction) { (match: Any, snippet: String?) in | ||
|  | 
 | ||
|  |             switch match { | ||
|  |             case let groupThread as TSGroupThread: | ||
|  |                 let sortKey = ConversationSortKey(creationDate: groupThread.creationDate, | ||
|  |                                                   lastMessageReceivedAtDate: groupThread.lastInteractionForInbox(transaction: transaction)?.receivedAtDate()) | ||
|  |                 let threadViewModel = ThreadViewModel(thread: groupThread, transaction: transaction) | ||
|  |                 let searchResult = GroupSearchResult(thread: threadViewModel, sortKey: sortKey) | ||
|  |                 groups.append(searchResult) | ||
|  |             case is TSContactThread: | ||
|  |                 // not included in compose screen results | ||
|  |                 break | ||
|  |             case is TSMessage: | ||
|  |                 // not included in compose screen results | ||
|  |                 break | ||
|  |             default: | ||
|  |                 owsFailDebug("unhandled item: \(match)") | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         // Order the conversation and message results in reverse chronological order. | ||
|  |         // The contact results are pre-sorted by display name. | ||
|  |         groups.sort(by: >) | ||
|  | 
 | ||
| 
											5 years ago
										 |         return ComposeScreenSearchResultSet(searchText: searchText, groups: groups) | ||
| 
											7 years ago
										 |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     public func searchForHomeScreen(searchText: String, | ||
| 
											5 years ago
										 |                                     transaction: YapDatabaseReadTransaction) -> HomeScreenSearchResultSet { | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |         var conversations: [ConversationSearchResult<ConversationSortKey>] = [] | ||
|  |         var messages: [ConversationSearchResult<MessageSortKey>] = [] | ||
| 
											7 years ago
										 | 
 | ||
|  |         var existingConversationRecipientIds: Set<String> = Set() | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |         self.finder.enumerateObjects(searchText: searchText, transaction: transaction) { (match: Any, snippet: String?) in | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |             if let thread = match as? TSThread { | ||
|  |                 let threadViewModel = ThreadViewModel(thread: thread, transaction: transaction) | ||
| 
											7 years ago
										 |                 let sortKey = ConversationSortKey(creationDate: thread.creationDate, | ||
|  |                                                   lastMessageReceivedAtDate: thread.lastInteractionForInbox(transaction: transaction)?.receivedAtDate()) | ||
| 
											7 years ago
										 |                 let searchResult = ConversationSearchResult(thread: threadViewModel, sortKey: sortKey) | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |                 if let contactThread = thread as? TSContactThread { | ||
|  |                     let recipientId = contactThread.contactIdentifier() | ||
|  |                     existingConversationRecipientIds.insert(recipientId) | ||
|  |                 } | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |                 conversations.append(searchResult) | ||
| 
											7 years ago
										 |             } else if let message = match as? TSMessage { | ||
|  |                 let thread = message.thread(with: transaction) | ||
|  | 
 | ||
|  |                 let threadViewModel = ThreadViewModel(thread: thread, transaction: transaction) | ||
| 
											7 years ago
										 |                 let sortKey = message.sortId | ||
| 
											7 years ago
										 |                 let searchResult = ConversationSearchResult(thread: threadViewModel, | ||
|  |                                                             sortKey: sortKey, | ||
|  |                                                             messageId: message.uniqueId, | ||
| 
											7 years ago
										 |                                                             messageDate: NSDate.ows_date(withMillisecondsSince1970: message.timestamp), | ||
| 
											7 years ago
										 |                                                             snippet: snippet) | ||
|  | 
 | ||
| 
											7 years ago
										 |                 messages.append(searchResult) | ||
| 
											7 years ago
										 |             } else { | ||
| 
											7 years ago
										 |                 owsFailDebug("unhandled item: \(match)") | ||
| 
											7 years ago
										 |             } | ||
|  |         } | ||
|  | 
 | ||
| 
											7 years ago
										 |         // Order the conversation and message results in reverse chronological order. | ||
|  |         // The contact results are pre-sorted by display name. | ||
| 
											7 years ago
										 |         conversations.sort(by: >) | ||
|  |         messages.sort(by: >) | ||
| 
											7 years ago
										 | 
 | ||
| 
											5 years ago
										 |         return HomeScreenSearchResultSet(searchText: searchText, conversations: conversations, messages: messages) | ||
| 
											7 years ago
										 |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     public func searchWithinConversation(thread: TSThread, | ||
|  |                                          searchText: String, | ||
|  |                                          transaction: YapDatabaseReadTransaction) -> ConversationScreenSearchResultSet { | ||
|  | 
 | ||
|  |         var messages: [MessageSearchResult] = [] | ||
|  | 
 | ||
|  |         guard let threadId = thread.uniqueId else { | ||
|  |             owsFailDebug("threadId was unexpectedly nil") | ||
|  |             return ConversationScreenSearchResultSet.empty | ||
|  |         } | ||
|  | 
 | ||
|  |         self.finder.enumerateObjects(searchText: searchText, transaction: transaction) { (match: Any, snippet: String?) in | ||
|  |             if let message = match as? TSMessage { | ||
|  |                 guard message.uniqueThreadId == threadId else { | ||
|  |                     return | ||
|  |                 } | ||
|  | 
 | ||
|  |                 guard let messageId = message.uniqueId else { | ||
|  |                     owsFailDebug("messageId was unexpectedly nil") | ||
|  |                     return | ||
|  |                 } | ||
|  | 
 | ||
|  |                 let searchResult = MessageSearchResult(messageId: messageId, sortId: message.sortId) | ||
|  |                 messages.append(searchResult) | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         // We want most recent first | ||
|  |         messages.sort(by: >) | ||
|  | 
 | ||
|  |         return ConversationScreenSearchResultSet(searchText: searchText, messages: messages) | ||
|  |     } | ||
|  | 
 | ||
| 
											8 years ago
										 |     @objc(filterThreads:withSearchText:) | ||
|  |     public func filterThreads(_ threads: [TSThread], searchText: String) -> [TSThread] { | ||
| 
											6 years ago
										 |         let threads = threads.filter { $0.name() != "Session Updates" && $0.name() != "Loki News" } | ||
| 
											8 years ago
										 |         guard searchText.trimmingCharacters(in: .whitespacesAndNewlines).count > 0 else { | ||
|  |             return threads | ||
|  |         } | ||
|  | 
 | ||
|  |         return threads.filter { thread in | ||
|  |             switch thread { | ||
|  |             case let groupThread as TSGroupThread: | ||
|  |                 return self.groupThreadSearcher.matches(item: groupThread, query: searchText) | ||
|  |             case let contactThread as TSContactThread: | ||
|  |                 return self.contactThreadSearcher.matches(item: contactThread, query: searchText) | ||
|  |             default: | ||
| 
											7 years ago
										 |                 owsFailDebug("Unexpected thread type: \(thread)") | ||
| 
											8 years ago
										 |                 return false | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     @objc(filterGroupThreads:withSearchText:) | ||
|  |     public func filterGroupThreads(_ groupThreads: [TSGroupThread], searchText: String) -> [TSGroupThread] { | ||
|  |         guard searchText.trimmingCharacters(in: .whitespacesAndNewlines).count > 0 else { | ||
|  |             return groupThreads | ||
|  |         } | ||
|  | 
 | ||
|  |         return groupThreads.filter { groupThread in | ||
|  |             return self.groupThreadSearcher.matches(item: groupThread, query: searchText) | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     @objc(filterSignalAccounts:withSearchText:) | ||
|  |     public func filterSignalAccounts(_ signalAccounts: [SignalAccount], searchText: String) -> [SignalAccount] { | ||
|  |         guard searchText.trimmingCharacters(in: .whitespacesAndNewlines).count > 0 else { | ||
|  |             return signalAccounts | ||
|  |         } | ||
|  | 
 | ||
|  |         return signalAccounts.filter { signalAccount in | ||
|  |             self.signalAccountSearcher.matches(item: signalAccount, query: searchText) | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Searchers | ||
| 
											7 years ago
										 | 
 | ||
| 
											8 years ago
										 |     private lazy var groupThreadSearcher: Searcher<TSGroupThread> = Searcher { (groupThread: TSGroupThread) in | ||
|  |         let groupName = groupThread.groupModel.groupName | ||
|  |         let memberStrings = groupThread.groupModel.groupMemberIds.map { recipientId in | ||
|  |             self.indexingString(recipientId: recipientId) | ||
| 
											7 years ago
										 |         }.joined(separator: " ") | ||
| 
											8 years ago
										 | 
 | ||
|  |         return "\(memberStrings) \(groupName ?? "")" | ||
|  |     } | ||
|  | 
 | ||
|  |     private lazy var contactThreadSearcher: Searcher<TSContactThread> = Searcher { (contactThread: TSContactThread) in | ||
|  |         let recipientId = contactThread.contactIdentifier() | ||
| 
											7 years ago
										 |         return self.conversationIndexingString(recipientId: recipientId) | ||
| 
											8 years ago
										 |     } | ||
|  | 
 | ||
|  |     private lazy var signalAccountSearcher: Searcher<SignalAccount> = Searcher { (signalAccount: SignalAccount) in | ||
|  |         let recipientId = signalAccount.recipientId | ||
| 
											7 years ago
										 |         return self.conversationIndexingString(recipientId: recipientId) | ||
|  |     } | ||
|  | 
 | ||
|  |     private func conversationIndexingString(recipientId: String) -> String { | ||
|  |         var result = self.indexingString(recipientId: recipientId) | ||
|  | 
 | ||
| 
											7 years ago
										 |         if IsNoteToSelfEnabled(), | ||
|  |             let localNumber = tsAccountManager.localNumber(), | ||
|  |             localNumber == recipientId { | ||
|  |             let noteToSelfLabel = NSLocalizedString("NOTE_TO_SELF", comment: "Label for 1:1 conversation with yourself.") | ||
|  |             result += " \(noteToSelfLabel)" | ||
| 
											7 years ago
										 |         } | ||
|  | 
 | ||
|  |         return result | ||
| 
											8 years ago
										 |     } | ||
|  | 
 | ||
|  |     private func indexingString(recipientId: String) -> String { | ||
| 
											5 years ago
										 |         let profileName = SSKEnvironment.shared.profileManager.profileNameForRecipient(withID: recipientId, avoidingWriteTransaction: true) | ||
| 
											8 years ago
										 | 
 | ||
| 
											5 years ago
										 |         return "\(recipientId) \(profileName ?? "")" | ||
| 
											8 years ago
										 |     } | ||
|  | } |