BasalProfileEditorStateModel.swift 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. import Observation
  2. import SwiftUI
  3. extension BasalProfileEditor {
  4. @Observable final class StateModel: BaseStateModel<Provider> {
  5. @ObservationIgnored @Injected() private var nightscout: NightscoutManager!
  6. var syncInProgress: Bool = false
  7. var initialItems: [Item] = []
  8. var items: [Item] = []
  9. var total: Decimal = 0.0
  10. var showAlert: Bool = false
  11. var chartData: [BasalProfile]? = []
  12. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  13. private(set) var rateValues: [Decimal] = []
  14. var canAdd: Bool {
  15. guard let lastItem = items.last else { return true }
  16. return lastItem.timeIndex < timeValues.count - 1
  17. }
  18. var hasChanges: Bool {
  19. initialItems != items
  20. }
  21. override func subscribe() {
  22. rateValues = provider.supportedBasalRates ?? stride(from: 5.0, to: 1001.0, by: 5.0)
  23. .map { ($0.decimal ?? .zero) / 100 }
  24. items = provider.profile.map { value in
  25. let timeIndex = timeValues.firstIndex(of: Double(value.minutes * 60)) ?? 0
  26. let rateIndex = rateValues.firstIndex(of: value.rate) ?? 0
  27. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  28. }
  29. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  30. calcTotal()
  31. }
  32. func calcTotal() {
  33. let profile = items.map { item -> BasalProfileEntry in
  34. let fotmatter = DateFormatter()
  35. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  36. fotmatter.dateFormat = "HH:mm:ss"
  37. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  38. let minutes = Int(date.timeIntervalSince1970 / 60)
  39. let rate = self.rateValues[item.rateIndex]
  40. return BasalProfileEntry(start: fotmatter.string(from: date), minutes: minutes, rate: rate)
  41. }
  42. var profileWith24hours = profile.map(\.minutes)
  43. profileWith24hours.append(24 * 60)
  44. let pr2 = zip(profile, profileWith24hours.dropFirst())
  45. total = pr2.reduce(0) { $0 + (Decimal($1.1 - $1.0.minutes) / 60) * $1.0.rate }
  46. }
  47. func add() {
  48. var time = 0
  49. var rate = 0
  50. if let last = items.last {
  51. time = last.timeIndex + 1
  52. rate = last.rateIndex
  53. }
  54. let newItem = Item(rateIndex: rate, timeIndex: time)
  55. items.append(newItem)
  56. calcTotal()
  57. }
  58. func save() {
  59. guard hasChanges else { return }
  60. syncInProgress = true
  61. let profile = items.map { item -> BasalProfileEntry in
  62. let formatter = DateFormatter()
  63. formatter.timeZone = TimeZone(secondsFromGMT: 0)
  64. formatter.dateFormat = "HH:mm:ss"
  65. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  66. let minutes = Int(date.timeIntervalSince1970 / 60)
  67. let rate = self.rateValues[item.rateIndex]
  68. return BasalProfileEntry(start: formatter.string(from: date), minutes: minutes, rate: rate)
  69. }
  70. provider.saveProfile(profile)
  71. .receive(on: DispatchQueue.main)
  72. .sink { completion in
  73. self.syncInProgress = false
  74. switch completion {
  75. case .finished:
  76. // Successfully saved and synced
  77. self.initialItems = self.items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  78. Task.detached(priority: .low) {
  79. debug(.nightscout, "Attempting to upload basal rates to Nightscout")
  80. await self.nightscout.uploadProfiles()
  81. }
  82. case .failure:
  83. // Handle the error, show error message
  84. self.showAlert = true
  85. }
  86. } receiveValue: {
  87. // Handle any successful value if needed
  88. print("We were successful")
  89. }
  90. .store(in: &lifetime)
  91. }
  92. func validate() {
  93. DispatchQueue.main.async {
  94. let uniq = Array(Set(self.items))
  95. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  96. sorted.first?.timeIndex = 0
  97. if self.items != sorted {
  98. self.items = sorted
  99. }
  100. self.calcTotal()
  101. }
  102. }
  103. func availableTimeIndices(_ itemIndex: Int) -> [Int] {
  104. // avoid index out of range issues
  105. guard itemIndex >= 0, itemIndex < items.count else {
  106. return []
  107. }
  108. let usedIndicesByOtherItems = items
  109. .enumerated()
  110. .filter { $0.offset != itemIndex }
  111. .map(\.element.timeIndex)
  112. return (0 ..< timeValues.count).filter { !usedIndicesByOtherItems.contains($0) }
  113. }
  114. func caluclateChartData() {
  115. DispatchQueue.main.async {
  116. var basals: [BasalProfile] = []
  117. let tzOffset = TimeZone.current.secondsFromGMT() * -1
  118. basals.append(contentsOf: self.items.enumerated().map { index, item in
  119. let startDate = Date(timeIntervalSinceReferenceDate: self.timeValues[item.timeIndex])
  120. var endDate = Date(timeIntervalSinceReferenceDate: self.timeValues.last!).addingTimeInterval(30 * 60)
  121. if self.items.count > index + 1 {
  122. let nextItem = self.items[index + 1]
  123. endDate = Date(timeIntervalSinceReferenceDate: self.timeValues[nextItem.timeIndex])
  124. }
  125. return BasalProfile(
  126. amount: Double(self.rateValues[item.rateIndex]),
  127. isOverwritten: false,
  128. startDate: startDate.addingTimeInterval(TimeInterval(tzOffset)),
  129. endDate: endDate.addingTimeInterval(TimeInterval(tzOffset))
  130. )
  131. })
  132. basals.sort(by: {
  133. $0.startDate > $1.startDate
  134. })
  135. self.chartData = basals
  136. }
  137. }
  138. }
  139. }