StateIntentRequest.swift 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. import AppIntents
  2. import Foundation
  3. enum StateIntentError: Error {
  4. case StateIntentUnknownError
  5. case NoBG
  6. case NoIOBCOB
  7. }
  8. @available(iOS 16, *) struct StateiAPSResults: AppEntity {
  9. static var defaultQuery = StateBGQuery()
  10. static var typeDisplayRepresentation: TypeDisplayRepresentation = "iAPS State Result"
  11. var displayRepresentation: DisplayRepresentation {
  12. DisplayRepresentation(title: "\(glucose)")
  13. }
  14. var id: UUID
  15. @Property(title: "Glucose") var glucose: String
  16. @Property(title: "Trend") var trend: String
  17. @Property(title: "Delta") var delta: String
  18. @Property(title: "Date") var date: Date
  19. @Property(title: "IOB") var iob: Double?
  20. @Property(title: "COB") var cob: Double?
  21. @Property(title: "unit") var unit: String?
  22. init(glucose: String, trend: String, delta: String, date: Date, iob: Double, cob: Double, unit: GlucoseUnits) {
  23. id = UUID()
  24. self.glucose = glucose
  25. self.trend = trend
  26. self.delta = delta
  27. self.date = date
  28. self.iob = iob
  29. self.cob = cob
  30. self.unit = unit.rawValue
  31. }
  32. }
  33. @available(iOS 16.0, *) struct StateBGQuery: EntityQuery {
  34. func entities(for _: [StateiAPSResults.ID]) async throws -> [StateiAPSResults] {
  35. []
  36. }
  37. func suggestedEntities() async throws -> [StateiAPSResults] {
  38. []
  39. }
  40. }
  41. @available(iOS 16.0, *) final class StateIntentRequest: BaseIntentsRequest {
  42. let moc = CoreDataStack.shared.persistentContainer.newBackgroundContext()
  43. func getLastGlucose() throws -> (dateGlucose: Date, glucose: String, trend: String, delta: String) {
  44. do {
  45. let results = CoreDataStack.shared.fetchEntities(
  46. ofType: GlucoseStored.self,
  47. predicate: NSPredicate.predicateFor30MinAgo,
  48. key: "date",
  49. ascending: false,
  50. fetchLimit: 2
  51. )
  52. guard let lastValue = results.first else { throw StateIntentError.NoBG }
  53. /// calculate delta
  54. let lastGlucose = lastValue.glucose
  55. let secondLastGlucose = results.dropFirst().first?.glucose
  56. let delta = results.count > 1 ? (lastGlucose - (secondLastGlucose ?? 0)) : nil
  57. /// formatting
  58. let units = settingsManager.settings.units
  59. let glucoseAsString = glucoseFormatter.string(from: Double(
  60. units == .mmolL ? Decimal(lastGlucose)
  61. .asMmolL : Decimal(lastGlucose)
  62. ) as NSNumber)!
  63. let directionAsString = lastValue.direction ?? "none"
  64. let deltaAsString = delta
  65. .map {
  66. self.deltaFormatter
  67. .string(from: Double(
  68. units == .mmolL ? Decimal($0)
  69. .asMmolL : Decimal($0)
  70. ) as NSNumber)!
  71. } ?? "--"
  72. debugPrint("StateIntentRequest: \(#function) \(DebuggingIdentifiers.succeeded) fetched latest 2 glucose values")
  73. return (lastValue.date ?? Date(), glucoseAsString, directionAsString, deltaAsString)
  74. } catch {
  75. debugPrint("StateIntentRequest: \(#function) \(DebuggingIdentifiers.failed) failed to fetch latest 2 glucose values")
  76. return (Date(), "", "", "")
  77. }
  78. }
  79. func getIobAndCob() throws -> (iob: Double, cob: Double) {
  80. let results = CoreDataStack.shared.fetchEntities(
  81. ofType: OrefDetermination.self,
  82. predicate: NSPredicate.enactedDetermination,
  83. key: "deliverAt",
  84. ascending: false,
  85. fetchLimit: 1
  86. )
  87. let iobAsDouble = Double(truncating: (results.first?.iob ?? 0.0) as NSNumber)
  88. let cobAsDouble = Double(truncating: (results.first?.cob ?? 0) as NSNumber)
  89. return (iobAsDouble, cobAsDouble)
  90. }
  91. private var glucoseFormatter: NumberFormatter {
  92. let formatter = NumberFormatter()
  93. formatter.numberStyle = .decimal
  94. formatter.maximumFractionDigits = 0
  95. if settingsManager.settings.units == .mmolL {
  96. formatter.minimumFractionDigits = 1
  97. formatter.maximumFractionDigits = 1
  98. }
  99. formatter.roundingMode = .halfUp
  100. return formatter
  101. }
  102. private var deltaFormatter: NumberFormatter {
  103. let formatter = NumberFormatter()
  104. formatter.numberStyle = .decimal
  105. formatter.maximumFractionDigits = 1
  106. formatter.positivePrefix = "+"
  107. return formatter
  108. }
  109. }