Meshtastic-Apple/Meshtastic/Extensions/CoreData/MessageEntityExtension.swift

87 lines
2.2 KiB
Swift
Raw Normal View History

2023-08-23 07:03:05 -05:00
//
// MessageEntityExtension.swift
// Meshtastic
//
// Created by Ben on 8/22/23.
//
import CoreData
import CoreLocation
2025-12-16 06:44:34 -08:00
import Foundation
2023-08-23 07:03:05 -05:00
import MapKit
import SwiftUI
extension MessageEntity {
var timestamp: Date {
2024-08-03 07:02:51 -07:00
let time = messageTimestamp
2023-08-23 07:03:05 -05:00
return Date(timeIntervalSince1970: TimeInterval(time))
}
var canRetry: Bool {
let re = RoutingError(rawValue: Int(ackError))
return re?.canRetry ?? false
}
2024-07-23 19:10:00 -07:00
var tapbacks: [MessageEntity] {
let context = PersistenceController.shared.container.viewContext
let fetchRequest = MessageEntity.fetchRequest()
2025-12-16 06:44:34 -08:00
fetchRequest.sortDescriptors = [
NSSortDescriptor(key: "messageTimestamp", ascending: true)
]
fetchRequest.predicate = NSPredicate(
format: "replyID == %lld AND isEmoji == true",
self.messageId
)
2024-07-23 19:10:00 -07:00
return (try? context.fetch(fetchRequest)) ?? [MessageEntity]()
}
func displayTimestamp(aboveMessage: MessageEntity?) -> Bool {
if let aboveMessage = aboveMessage {
2025-12-16 06:44:34 -08:00
return aboveMessage.timestamp.addingTimeInterval(3600) < timestamp // 60 minutes
}
2025-12-16 06:44:34 -08:00
return false // First message will have no timestamp
}
2025-12-16 06:44:34 -08:00
func relayDisplay() -> String? {
guard self.relayNode != 0 else { return nil }
let context = PersistenceController.shared.container.viewContext
let relaySuffix = Int64(self.relayNode & 0xFF)
let request: NSFetchRequest<UserEntity> = UserEntity.fetchRequest()
request.predicate = NSPredicate(
format: "(num & 0xFF) == %lld",
relaySuffix
)
do {
let users = try context.fetch(request)
// If exactly one match is found, return its name
if users.count == 1, let name = users.first?.longName, !name.isEmpty
{
return "\(name)"
}
2025-12-16 06:44:34 -08:00
// If no exact match, find the node with the smallest hopsAway
if let closestNode = users.min(by: { lhs, rhs in
guard let lhsHops = lhs.userNode?.hopsAway,
let rhsHops = rhs.userNode?.hopsAway
else {
return false
}
return lhsHops < rhsHops
}), let name = closestNode.longName, !name.isEmpty {
return "\(name)"
}
2025-12-16 06:44:34 -08:00
// Fallback to hex node number if no matches
return String(format: "Node 0x%02X", UInt32(self.relayNode & 0xFF))
2025-12-16 06:44:34 -08:00
} catch {
return String(format: "Node 0x%02X", UInt32(self.relayNode & 0xFF))
}
}
2023-08-23 07:03:05 -05:00
}