ISFEditorStateModel.swift 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  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. var items: [Item] = []
  17. var initialItems: [Item] = []
  18. var shouldDisplaySaving: Bool = false
  19. let context = CoreDataStack.shared.newTaskContext()
  20. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  21. var rateValues: [Decimal] {
  22. var values = stride(from: 9, to: 540.01, by: 1.0).map { Decimal($0) }
  23. if units == .mmolL {
  24. var mmolValues = values.filter { Int(truncating: $0 as NSNumber) % 2 == 0 }
  25. // check for any missing values
  26. var valuesInMmolSet = Set(mmolValues.map(\.asMmolL))
  27. for value in values {
  28. let valueInMmmol = value.asMmolL
  29. if valuesInMmolSet.insert(valueInMmmol).inserted {
  30. mmolValues.append(value)
  31. }
  32. }
  33. values = mmolValues.sorted()
  34. }
  35. return values
  36. }
  37. var canAdd: Bool {
  38. guard let lastItem = items.last else { return true }
  39. return lastItem.timeIndex < timeValues.count - 1
  40. }
  41. var hasChanges: Bool {
  42. initialItems != items
  43. }
  44. private(set) var units: GlucoseUnits = .mgdL
  45. override func subscribe() {
  46. units = settingsManager.settings.units
  47. let profile = provider.profile
  48. items = profile.sensitivities.map { value in
  49. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  50. var rateIndex = rateValues.firstIndex(of: value.sensitivity)
  51. if rateIndex == nil {
  52. // try to look up the closest value
  53. if let min = rateValues.first, let max = rateValues.last {
  54. if value.sensitivity >= (min - 1), value.sensitivity <= (max + 1) {
  55. rateIndex = rateValues.findClosestIndex(to: value.sensitivity)
  56. }
  57. }
  58. }
  59. return Item(rateIndex: rateIndex ?? 0, timeIndex: timeIndex)
  60. }
  61. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  62. }
  63. func add() {
  64. var time = 0
  65. var rate = 0
  66. if let last = items.last {
  67. time = last.timeIndex + 1
  68. rate = last.rateIndex
  69. }
  70. let newItem = Item(rateIndex: rate, timeIndex: time)
  71. items.append(newItem)
  72. }
  73. func save() {
  74. guard hasChanges else { return }
  75. shouldDisplaySaving.toggle()
  76. let sensitivities = items.map { item -> InsulinSensitivityEntry in
  77. let fotmatter = DateFormatter()
  78. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  79. fotmatter.dateFormat = "HH:mm:ss"
  80. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  81. let minutes = Int(date.timeIntervalSince1970 / 60)
  82. let rate = self.rateValues[item.rateIndex]
  83. return InsulinSensitivityEntry(sensitivity: rate, offset: minutes, start: fotmatter.string(from: date))
  84. }
  85. let profile = InsulinSensitivities(
  86. units: .mgdL,
  87. userPreferredUnits: .mgdL,
  88. sensitivities: sensitivities
  89. )
  90. provider.saveProfile(profile)
  91. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  92. Task.detached(priority: .low) {
  93. do {
  94. debug(.nightscout, "Attempting to upload ISF to Nightscout")
  95. try await self.nightscout.uploadProfiles()
  96. } catch {
  97. debug(
  98. .default,
  99. "\(DebuggingIdentifiers.failed) Faile to upload ISF to Nightscout: \(error.localizedDescription)"
  100. )
  101. }
  102. }
  103. }
  104. func validate() {
  105. DispatchQueue.main.async {
  106. DispatchQueue.main.async {
  107. let uniq = Array(Set(self.items))
  108. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  109. sorted.first?.timeIndex = 0
  110. if self.items != sorted {
  111. self.items = sorted
  112. }
  113. if self.items.isEmpty {
  114. self.units = self.settingsManager.settings.units
  115. }
  116. }
  117. }
  118. }
  119. }
  120. }
  121. extension ISFEditor.StateModel: SettingsObserver {
  122. func settingsDidChange(_: TrioSettings) {
  123. units = settingsManager.settings.units
  124. }
  125. }