// // UserEntityExtension.swift // Meshtastic // // Copyright(c) Garth Vander Houwen 6/3/22. // import Foundation import CoreData import MeshtasticProtobufs extension UserEntity { var messagePredicate: NSPredicate { return NSPredicate(format: "((toUser == %@) OR (fromUser == %@)) AND toUser != nil AND fromUser != nil AND isEmoji == false AND admin = false AND portNum != 10", self, self) } var messageFetchRequest: NSFetchRequest { let fetchRequest = MessageEntity.fetchRequest() fetchRequest.sortDescriptors = [NSSortDescriptor(key: "messageTimestamp", ascending: true)] fetchRequest.predicate = messagePredicate return fetchRequest } var messageList: [MessageEntity] { let context = PersistenceController.shared.container.viewContext let fetchRequest = messageFetchRequest return (try? context.fetch(fetchRequest)) ?? [MessageEntity]() } var mostRecentMessage: MessageEntity? { // Most contacts will have no DMs history, so we can return early. guard self.lastMessage != nil else { return nil; } // Most recent DM for this user (descending, limit 1) let context = PersistenceController.shared.container.viewContext let fetchRequest = messageFetchRequest fetchRequest.sortDescriptors = [NSSortDescriptor(key: "messageTimestamp", ascending: false)] fetchRequest.fetchLimit = 1 return (try? context.fetch(fetchRequest))?.first } var sensorMessageList: [MessageEntity] { let context = PersistenceController.shared.container.viewContext let fetchRequest = MessageEntity.fetchRequest() fetchRequest.sortDescriptors = [NSSortDescriptor(key: "messageTimestamp", ascending: true)] fetchRequest.predicate = NSPredicate(format: "(fromUser == %@) AND portNum = 10", self) return (try? context.fetch(fetchRequest)) ?? [MessageEntity]() } func unreadMessages(context: NSManagedObjectContext, skipLastMessageCheck: Bool = false) -> Int { // Most contacts will have no DMs history, so we can return early. // (For our own node, set skipLastMessageCheck=true, because we don't update lastMessage on our own connected node.) guard self.lastMessage != nil || skipLastMessageCheck else { return 0; } let fetchRequest = messageFetchRequest fetchRequest.sortDescriptors = [] // sort is irrelevant. fetchRequest.predicate = NSCompoundPredicate(andPredicateWithSubpredicates: [fetchRequest.predicate!, NSPredicate(format: "read == false")]) return (try? context.count(for: fetchRequest)) ?? 0 } // Backwards-compatible property (uses viewContext) var unreadMessages: Int { unreadMessages(context: PersistenceController.shared.container.viewContext) } /// SVG Images for Vendors who are signed project backers var hardwareImage: String? { guard let hwModel else { return nil } switch hwModel { /// Heltec case "HELTECHT62": return "HELTECHT62" case "HELTECMESHNODET114": return "HELTECMESHNODET114" case "HELTECV3": return "HELTECV3" case "HELTECMESHPOCKET": return "HELTECMESHPOCKET" case "HELTECVISIONMASTERE213": return "HELTECVISIONMASTERE213" case "HELTECVISIONMASTERE290": return "HELTECVISIONMASTERE290" case "HELTECWIRELESSPAPER", "HELTECWIRELESSPAPERV10": return "HELTECWIRELESSPAPER" case "HELTECWIRELESSTRACKER", "HELTECWIRELESSTRACKERV10": return "HELTECWIRELESSTRACKER" case "HELTECWSLV3": return "HELTECWSLV3" /// LilyGO case "TDECK": return "TDECK" case "TECHO": return "TECHO" case "TWATCHS3": return "TWATCHS3" case "LILYGOTBEAMS3CORE": return "LILYGOTBEAMS3CORE" case "TBEAM", "TBEAM_V0P7": return "TBEAM" case "TLORAC6": return "TLORAC6" case "TLORAT3S3EPAPER": return "TLORAT3S3EPAPER" case "TLORAT3S3V1", "TLORAT3S3": return "TLORAT3S3V1" case "TLORAV211P6": return "TLORAV211P6" case "TLORAV211P8": return "TLORAV211P8" /// Seeed Studio case "SENSECAPINDICATOR": return "SENSECAPINDICATOR" case "TRACKERT1000E": return "TRACKERT1000E" case "SEEEDXIAOS3": return "SEEEDXIAOS3" case "WIOWM1110": return "WIOWM1110" case "SEEEDSOLARNODE": return "SEEEDSOLARNODE" case "SEEEDWIOTRACKERL1": return "SEEEDWIOTRACKERL1" /// RAK Wireless case "RAK4631": return "RAK4631" case "RAK11310": return "RAK11310" case "WISMESHTAP": return "WISMESHTAP" /// B&Q Consulting case "NANOG1", "NANOG1EXPLORER": return "NANOG1" case "NANOG2ULTRA": return "NANOG2ULTRA" /// Muzi Works case "MUZIR1NEO": return "MUZIR1NEO" case "STATIONG2": return "STATIONG2" /// DIY Devices case "RPIPICO": return "RPIPICO" default: return "UNSET" } } } public func createUser(num: Int64, context: NSManagedObjectContext) throws -> UserEntity { // Validate Input guard num >= 0 else { throw CoreDataError.invalidInput(message: "User number cannot be negative.") } var newUser: UserEntity! // Use an implicitly unwrapped optional, but ensure it's assigned context.performAndWait { newUser = UserEntity(context: context) newUser.num = num let userId = num.toHex() newUser.userId = userId let last4 = String(userId.suffix(4)) newUser.longName = "Meshtastic \(last4)" newUser.shortName = last4 newUser.hwModel = "UNSET" newUser.unmessagable = false } return newUser } enum CoreDataError: Error, LocalizedError { case invalidInput(message: String) case saveFailed(message: String) case entityCreationFailed(message: String) // In case UserEntity(context:) fails for some reason var errorDescription: String? { switch self { case .invalidInput(let message): return "Core Data Input Error: \(message)" case .saveFailed(let message): return "Core Data Save Error: \(message)" case .entityCreationFailed(let message): return "Core Data Entity Creation Error: \(message)" } } }