ISFEditorStateModel.swift 5.3 KB

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