TargetsEditorStateModel.swift 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. import SwiftUI
  2. extension TargetsEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Injected() private var nightscout: NightscoutManager!
  5. @Injected() private var tidepoolManager: TidepoolManager!
  6. @Injected() private var broadcaster: Broadcaster!
  7. @Published var items: [Item] = []
  8. @Published var initialItems: [Item] = []
  9. @Published var therapyItems: [TherapySettingItem] = []
  10. @Published var shouldDisplaySaving: Bool = false
  11. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  12. var rateValues: [Decimal] {
  13. let settingsProvider = PickerSettingsProvider.shared
  14. let glucoseSetting = PickerSetting(value: 110, step: 1, min: 72, max: 180, type: .glucose)
  15. return settingsProvider.generatePickerValues(from: glucoseSetting, units: units)
  16. }
  17. var canAdd: Bool {
  18. guard let lastItem = items.last else { return true }
  19. return lastItem.timeIndex < timeValues.count - 1
  20. }
  21. var hasChanges: Bool {
  22. initialItems != items
  23. }
  24. private(set) var units: GlucoseUnits = .mgdL
  25. // Convert items to TherapySettingItem format
  26. func getTherapyItems() -> [TherapySettingItem] {
  27. items.map { item in
  28. TherapySettingItem(
  29. time: timeValues[item.timeIndex],
  30. value: rateValues[item.lowIndex]
  31. )
  32. }
  33. }
  34. // Update items from TherapySettingItem format
  35. func updateFromTherapyItems(_ therapyItems: [TherapySettingItem]) {
  36. items = therapyItems.map { therapyItem in
  37. let timeIndex = timeValues.firstIndex(where: { abs($0 - therapyItem.time) < 1 }) ?? 0
  38. let lowIndex = rateValues.firstIndex(of: therapyItem.value) ?? 0
  39. return Item(lowIndex: lowIndex, highIndex: lowIndex, timeIndex: timeIndex)
  40. }
  41. }
  42. override func subscribe() {
  43. units = settingsManager.settings.units
  44. let profile = provider.profile
  45. items = profile.targets.map { value in
  46. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  47. let lowIndex = rateValues.firstIndex(of: value.low) ?? 0
  48. let highIndex = rateValues.firstIndex(of: value.high) ?? 0
  49. return Item(lowIndex: lowIndex, highIndex: highIndex, timeIndex: timeIndex)
  50. }
  51. initialItems = items.map { Item(lowIndex: $0.lowIndex, highIndex: $0.highIndex, timeIndex: $0.timeIndex) }
  52. }
  53. func add() {
  54. var time = 0
  55. var low = 0
  56. var high = 0
  57. if let last = items.last {
  58. time = last.timeIndex + 1
  59. low = last.lowIndex
  60. high = low
  61. }
  62. let newItem = Item(lowIndex: low, highIndex: high, timeIndex: time)
  63. items.append(newItem)
  64. }
  65. func save() {
  66. guard hasChanges else { return }
  67. shouldDisplaySaving.toggle()
  68. let targets = items.map { item -> BGTargetEntry in
  69. let formatter = DateFormatter()
  70. formatter.timeZone = TimeZone(secondsFromGMT: 0)
  71. formatter.dateFormat = "HH:mm:ss"
  72. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  73. let minutes = Int(date.timeIntervalSince1970 / 60)
  74. let low = self.rateValues[item.lowIndex]
  75. let high = low
  76. return BGTargetEntry(low: low, high: high, start: formatter.string(from: date), offset: minutes)
  77. }
  78. let profile = BGTargets(units: .mgdL, userPreferredUnits: .mgdL, targets: targets)
  79. provider.saveProfile(profile)
  80. initialItems = items.map { Item(lowIndex: $0.lowIndex, highIndex: $0.highIndex, timeIndex: $0.timeIndex) }
  81. DispatchQueue.main.async {
  82. self.broadcaster.notify(BGTargetsObserver.self, on: .main) {
  83. $0.bgTargetsDidChange(profile)
  84. }
  85. }
  86. Task.detached(priority: .low) {
  87. do {
  88. debug(.nightscout, "Attempting to upload targets to Nightscout")
  89. try await self.nightscout.uploadProfiles()
  90. } catch {
  91. debug(
  92. .default,
  93. "\(DebuggingIdentifiers.failed) failed to upload targets to Nightscout: \(error)"
  94. )
  95. }
  96. }
  97. Task.detached(priority: .low) {
  98. await self.tidepoolManager.uploadSettings()
  99. }
  100. }
  101. func validate() {
  102. DispatchQueue.main.async {
  103. let uniq = Array(Set(self.items))
  104. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  105. .map { item -> Item in
  106. Item(lowIndex: item.lowIndex, highIndex: item.highIndex, timeIndex: item.timeIndex)
  107. }
  108. sorted.first?.timeIndex = 0
  109. self.items = sorted
  110. if self.items.isEmpty {
  111. self.units = self.settingsManager.settings.units
  112. }
  113. }
  114. }
  115. }
  116. }