Meshtastic-Apple/Meshtastic/Extensions/CoreData/UserEntityExtension.swift
2025-12-21 09:24:50 -08:00

200 lines
5.9 KiB
Swift

//
// 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<MessageEntity> {
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 "HELTECV4":
return "HELTECV4"
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"
/// Elecrow
case "THINKNODEM1":
return "THINKNODEM1"
case "THINKNODEM2":
return "THINKNODEM2"
case "THINKNODEM3":
return "THINKNODEM3"
case "THINKNODEM3":
return "THINKNODEM4"
/// 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)"
}
}
}