CarbRatioEditorStateModel.swift 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. import SwiftUI
  2. extension CarbRatioEditor {
  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. let rateValues = stride(from: 10.0, to: 501.0, by: 1.0).map { ($0.decimal ?? .zero) / 10 }
  13. var canAdd: Bool {
  14. guard let lastItem = items.last else { return true }
  15. return lastItem.timeIndex < timeValues.count - 1
  16. }
  17. var hasChanges: Bool {
  18. if initialItems.count != items.count {
  19. return true
  20. }
  21. for (initialItem, currentItem) in zip(initialItems, items) {
  22. if initialItem.rateIndex != currentItem.rateIndex || initialItem.timeIndex != currentItem.timeIndex {
  23. return true
  24. }
  25. }
  26. return false
  27. }
  28. // Convert items to TherapySettingItem format
  29. func getTherapyItems() -> [TherapySettingItem] {
  30. items.map { item in
  31. TherapySettingItem(
  32. time: timeValues[item.timeIndex],
  33. value: rateValues[item.rateIndex]
  34. )
  35. }
  36. }
  37. // Update items from TherapySettingItem format
  38. func updateFromTherapyItems(_ therapyItems: [TherapySettingItem]) {
  39. items = therapyItems.map { therapyItem in
  40. let timeIndex = timeValues.firstIndex(where: { abs($0 - therapyItem.time) < 1 }) ?? 0
  41. let rateIndex = rateValues.firstIndex(of: therapyItem.value) ?? 0
  42. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  43. }
  44. }
  45. override func subscribe() {
  46. items = provider.profile.schedule.map { value in
  47. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  48. let rateIndex = rateValues.firstIndex(of: value.ratio) ?? 0
  49. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  50. }
  51. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  52. }
  53. func add() {
  54. var time = 0
  55. var rate = 0
  56. if let last = items.last {
  57. time = last.timeIndex + 1
  58. rate = last.rateIndex
  59. }
  60. let newItem = Item(rateIndex: rate, timeIndex: time)
  61. items.append(newItem)
  62. }
  63. func save() {
  64. guard hasChanges else { return }
  65. shouldDisplaySaving = true
  66. let schedule = items.enumerated().map { _, item -> CarbRatioEntry in
  67. let fotmatter = DateFormatter()
  68. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  69. fotmatter.dateFormat = "HH:mm:ss"
  70. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  71. let minutes = Int(date.timeIntervalSince1970 / 60)
  72. let rate = self.rateValues[item.rateIndex]
  73. return CarbRatioEntry(start: fotmatter.string(from: date), offset: minutes, ratio: rate)
  74. }
  75. let profile = CarbRatios(units: .grams, schedule: schedule)
  76. provider.saveProfile(profile)
  77. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  78. DispatchQueue.main.async {
  79. self.broadcaster.notify(CarbRatiosObserver.self, on: .main) {
  80. $0.carbRatiosDidChange(profile)
  81. }
  82. }
  83. Task.detached(priority: .low) {
  84. do {
  85. debug(.nightscout, "Attempting to upload CRs to Nightscout")
  86. try await self.nightscout.uploadProfiles()
  87. } catch {
  88. debug(.default, "Failed to upload CRs to Nightscout: \(error)")
  89. }
  90. }
  91. Task.detached(priority: .low) {
  92. await self.tidepoolManager.uploadSettings()
  93. }
  94. }
  95. func validate() {
  96. DispatchQueue.main.async {
  97. let uniq = Array(Set(self.items))
  98. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  99. sorted.first?.timeIndex = 0
  100. if self.items != sorted {
  101. self.items = sorted
  102. }
  103. }
  104. }
  105. }
  106. }