ISFEditorStateModel.swift 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  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. var autotune: Autotune?
  12. let context = CoreDataStack.shared.newTaskContext()
  13. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  14. var rateValues: [Decimal] {
  15. var values = stride(from: 9, to: 540.01, by: 1.0).map { Decimal($0) }
  16. if units == .mmolL {
  17. values = values.filter { Int(truncating: $0 as NSNumber) % 2 == 0 }
  18. }
  19. return values
  20. }
  21. var canAdd: Bool {
  22. guard let lastItem = items.last else { return true }
  23. return lastItem.timeIndex < timeValues.count - 1
  24. }
  25. var hasChanges: Bool {
  26. initialItems != items
  27. }
  28. private(set) var units: GlucoseUnits = .mgdL
  29. override func subscribe() {
  30. units = settingsManager.settings.units
  31. let profile = provider.profile
  32. items = profile.sensitivities.map { value in
  33. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  34. let rateIndex = rateValues.firstIndex(of: value.sensitivity) ?? 0
  35. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  36. }
  37. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  38. autotune = provider.autotune
  39. }
  40. func add() {
  41. var time = 0
  42. var rate = 0
  43. if let last = items.last {
  44. time = last.timeIndex + 1
  45. rate = last.rateIndex
  46. }
  47. let newItem = Item(rateIndex: rate, timeIndex: time)
  48. items.append(newItem)
  49. }
  50. func save() {
  51. guard hasChanges else { return }
  52. shouldDisplaySaving.toggle()
  53. let sensitivities = items.map { item -> InsulinSensitivityEntry in
  54. let fotmatter = DateFormatter()
  55. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  56. fotmatter.dateFormat = "HH:mm:ss"
  57. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  58. let minutes = Int(date.timeIntervalSince1970 / 60)
  59. let rate = self.rateValues[item.rateIndex]
  60. return InsulinSensitivityEntry(sensitivity: rate, offset: minutes, start: fotmatter.string(from: date))
  61. }
  62. let profile = InsulinSensitivities(
  63. units: .mgdL,
  64. userPreferredUnits: .mgdL,
  65. sensitivities: sensitivities
  66. )
  67. provider.saveProfile(profile)
  68. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  69. Task.detached(priority: .low) {
  70. debug(.nightscout, "Attempting to upload ISF to Nightscout")
  71. await self.nightscout.uploadProfiles()
  72. }
  73. }
  74. func validate() {
  75. DispatchQueue.main.async {
  76. DispatchQueue.main.async {
  77. let uniq = Array(Set(self.items))
  78. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  79. sorted.first?.timeIndex = 0
  80. if self.items != sorted {
  81. self.items = sorted
  82. }
  83. if self.items.isEmpty {
  84. self.units = self.settingsManager.settings.units
  85. }
  86. }
  87. }
  88. }
  89. }
  90. }
  91. extension ISFEditor.StateModel: SettingsObserver {
  92. func settingsDidChange(_: FreeAPSSettings) {
  93. units = settingsManager.settings.units
  94. }
  95. }