ISFEditorStateModel.swift 5.0 KB

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