ISFEditorStateModel.swift 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. import CoreData
  2. import SwiftUI
  3. extension ISFEditor {
  4. final class StateModel: BaseStateModel<Provider> {
  5. @Published var items: [Item] = []
  6. private(set) var autosensISF: Decimal?
  7. private(set) var autosensRatio: Decimal = 0
  8. @Published var autotune: Autotune?
  9. @Published var determinationsFromPersistence: [OrefDetermination] = []
  10. let context = CoreDataStack.shared.newTaskContext()
  11. let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  12. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  13. var rateValues: [Decimal] {
  14. switch units {
  15. case .mgdL:
  16. return stride(from: 9, to: 540.01, by: 1.0).map { Decimal($0) }
  17. case .mmolL:
  18. return stride(from: 1.0, to: 301.0, by: 1.0).map { ($0.decimal ?? .zero) / 10 }
  19. }
  20. }
  21. var canAdd: Bool {
  22. guard let lastItem = items.last else { return true }
  23. return lastItem.timeIndex < timeValues.count - 1
  24. }
  25. private(set) var units: GlucoseUnits = .mmolL
  26. override func subscribe() {
  27. let profile = provider.profile
  28. units = profile.units
  29. items = profile.sensitivities.map { value in
  30. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  31. let rateIndex = rateValues.firstIndex(of: value.sensitivity) ?? 0
  32. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  33. }
  34. autotune = provider.autotune
  35. if let newISF = provider.autosense.newisf {
  36. switch units {
  37. case .mgdL:
  38. autosensISF = newISF
  39. case .mmolL:
  40. autosensISF = newISF * GlucoseUnits.exchangeRate
  41. }
  42. }
  43. autosensRatio = provider.autosense.ratio
  44. setupDeterminationsArray()
  45. }
  46. func add() {
  47. var time = 0
  48. var rate = 0
  49. if let last = items.last {
  50. time = last.timeIndex + 1
  51. rate = last.rateIndex
  52. }
  53. let newItem = Item(rateIndex: rate, timeIndex: time)
  54. items.append(newItem)
  55. }
  56. func save() {
  57. let sensitivities = items.map { item -> InsulinSensitivityEntry in
  58. let fotmatter = DateFormatter()
  59. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  60. fotmatter.dateFormat = "HH:mm:ss"
  61. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  62. let minutes = Int(date.timeIntervalSince1970 / 60)
  63. let rate = self.rateValues[item.rateIndex]
  64. return InsulinSensitivityEntry(sensitivity: rate, offset: minutes, start: fotmatter.string(from: date))
  65. }
  66. let profile = InsulinSensitivities(
  67. units: units,
  68. userPrefferedUnits: settingsManager.settings.units,
  69. sensitivities: sensitivities
  70. )
  71. provider.saveProfile(profile)
  72. }
  73. func validate() {
  74. DispatchQueue.main.async {
  75. let uniq = Array(Set(self.items))
  76. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  77. sorted.first?.timeIndex = 0
  78. self.items = sorted
  79. if self.items.isEmpty {
  80. self.units = self.settingsManager.settings.units
  81. }
  82. }
  83. }
  84. private func setupDeterminationsArray() {
  85. Task {
  86. let ids = await self.fetchDeterminations()
  87. await updateDeterminationsArray(with: ids)
  88. }
  89. }
  90. private func fetchDeterminations() async -> [NSManagedObjectID] {
  91. CoreDataStack.shared.fetchEntities(
  92. ofType: OrefDetermination.self,
  93. onContext: context,
  94. predicate: NSPredicate.enactedDetermination,
  95. key: "deliverAt",
  96. ascending: false,
  97. fetchLimit: 1
  98. ).map(\.objectID)
  99. }
  100. @MainActor private func updateDeterminationsArray(with IDs: [NSManagedObjectID]) {
  101. do {
  102. let objects = try IDs.compactMap { id in
  103. try viewContext.existingObject(with: id) as? OrefDetermination
  104. }
  105. determinationsFromPersistence = objects
  106. } catch {
  107. debugPrint(
  108. "Home State: \(#function) \(DebuggingIdentifiers.failed) error while updating the glucose array: \(error.localizedDescription)"
  109. )
  110. }
  111. }
  112. }
  113. }