ISFEditorStateModel.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. import CoreData
  2. import Observation
  3. import SwiftUI
  4. extension [Decimal] {
  5. func findClosestIndex(to target: Element) -> Int? {
  6. guard !isEmpty else { return nil }
  7. return enumerated().min(by: {
  8. abs($0.element - target) < abs($1.element - target)
  9. })?.offset
  10. }
  11. }
  12. extension ISFEditor {
  13. @Observable final class StateModel: BaseStateModel<Provider> {
  14. @ObservationIgnored @Injected() var determinationStorage: DeterminationStorage!
  15. @ObservationIgnored @Injected() private var nightscout: NightscoutManager!
  16. @ObservationIgnored @Injected() private var tidepoolManager: TidepoolManager!
  17. @ObservationIgnored @Injected() private var broadcaster: Broadcaster!
  18. var items: [Item] = []
  19. var initialItems: [Item] = []
  20. var therapyItems: [TherapySettingItem] = []
  21. var shouldDisplaySaving: Bool = false
  22. let context = CoreDataStack.shared.newTaskContext()
  23. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  24. var rateValues: [Decimal] {
  25. let settingsProvider = PickerSettingsProvider.shared
  26. let sensitivityPickerSetting = PickerSetting(value: 100, step: 1, min: 9, max: 540, type: .glucose)
  27. return settingsProvider.generatePickerValues(from: sensitivityPickerSetting, units: units)
  28. }
  29. var canAdd: Bool {
  30. guard let lastItem = items.last else { return true }
  31. return lastItem.timeIndex < timeValues.count - 1
  32. }
  33. var hasChanges: Bool {
  34. initialItems != items
  35. }
  36. private(set) var units: GlucoseUnits = .mgdL
  37. // Convert items to TherapySettingItem format
  38. func getTherapyItems() -> [TherapySettingItem] {
  39. items.map { item in
  40. TherapySettingItem(
  41. time: timeValues[item.timeIndex],
  42. value: rateValues[item.rateIndex]
  43. )
  44. }
  45. }
  46. // Update items from TherapySettingItem format
  47. func updateFromTherapyItems(_ therapyItems: [TherapySettingItem]) {
  48. items = therapyItems.map { therapyItem in
  49. let timeIndex = timeValues.firstIndex(where: { abs($0 - therapyItem.time) < 1 }) ?? 0
  50. let rateIndex = rateValues.firstIndex(of: therapyItem.value) ?? 0
  51. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  52. }
  53. }
  54. override func subscribe() {
  55. units = settingsManager.settings.units
  56. let profile = provider.profile
  57. items = profile.sensitivities.map { value in
  58. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  59. var rateIndex = rateValues.firstIndex(of: value.sensitivity)
  60. if rateIndex == nil {
  61. // try to look up the closest value
  62. if let min = rateValues.first, let max = rateValues.last {
  63. if value.sensitivity >= (min - 1), value.sensitivity <= (max + 1) {
  64. rateIndex = rateValues.findClosestIndex(to: value.sensitivity)
  65. }
  66. }
  67. }
  68. return Item(rateIndex: rateIndex ?? 0, timeIndex: timeIndex)
  69. }
  70. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  71. }
  72. func add() {
  73. var time = 0
  74. var rate = 0
  75. if let last = items.last {
  76. time = last.timeIndex + 1
  77. rate = last.rateIndex
  78. }
  79. let newItem = Item(rateIndex: rate, timeIndex: time)
  80. items.append(newItem)
  81. }
  82. func save() {
  83. guard hasChanges else { return }
  84. shouldDisplaySaving.toggle()
  85. let sensitivities = items.map { item -> InsulinSensitivityEntry in
  86. let fotmatter = DateFormatter()
  87. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  88. fotmatter.dateFormat = "HH:mm:ss"
  89. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  90. let minutes = Int(date.timeIntervalSince1970 / 60)
  91. let rate = self.rateValues[item.rateIndex]
  92. return InsulinSensitivityEntry(sensitivity: rate, offset: minutes, start: fotmatter.string(from: date))
  93. }
  94. let profile = InsulinSensitivities(
  95. units: .mgdL,
  96. userPreferredUnits: .mgdL,
  97. sensitivities: sensitivities
  98. )
  99. provider.saveProfile(profile)
  100. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  101. DispatchQueue.main.async {
  102. self.broadcaster.notify(InsulinSensitivitiesObserver.self, on: .main) {
  103. $0.insulinSensitivitiesDidChange(profile)
  104. }
  105. }
  106. Task.detached(priority: .low) {
  107. do {
  108. debug(.nightscout, "Attempting to upload ISF to Nightscout")
  109. try await self.nightscout.uploadProfiles()
  110. } catch {
  111. debug(
  112. .default,
  113. "\(DebuggingIdentifiers.failed) Faile to upload ISF to Nightscout: \(error)"
  114. )
  115. }
  116. }
  117. Task.detached(priority: .low) {
  118. await self.tidepoolManager.uploadSettings()
  119. }
  120. }
  121. func validate() {
  122. DispatchQueue.main.async {
  123. DispatchQueue.main.async {
  124. let uniq = Array(Set(self.items))
  125. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  126. sorted.first?.timeIndex = 0
  127. if self.items != sorted {
  128. self.items = sorted
  129. }
  130. if self.items.isEmpty {
  131. self.units = self.settingsManager.settings.units
  132. }
  133. }
  134. }
  135. }
  136. }
  137. }
  138. extension ISFEditor.StateModel: SettingsObserver {
  139. func settingsDidChange(_: TrioSettings) {
  140. units = settingsManager.settings.units
  141. }
  142. }