BasalProfileEditorStateModel.swift 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. import SwiftUI
  2. extension BasalProfileEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Injected() private var nightscout: NightscoutManager!
  5. @Published var syncInProgress: Bool = false
  6. @Published var initialItems: [Item] = []
  7. @Published var items: [Item] = []
  8. @Published var total: Decimal = 0.0
  9. @Published var showAlert: Bool = false
  10. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  11. private(set) var rateValues: [Decimal] = []
  12. var canAdd: Bool {
  13. guard let lastItem = items.last else { return true }
  14. return lastItem.timeIndex < timeValues.count - 1
  15. }
  16. var hasChanges: Bool {
  17. initialItems != items
  18. }
  19. override func subscribe() {
  20. rateValues = provider.supportedBasalRates ?? stride(from: 5.0, to: 1001.0, by: 5.0)
  21. .map { ($0.decimal ?? .zero) / 100 }
  22. items = provider.profile.map { value in
  23. let timeIndex = timeValues.firstIndex(of: Double(value.minutes * 60)) ?? 0
  24. let rateIndex = rateValues.firstIndex(of: value.rate) ?? 0
  25. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  26. }
  27. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  28. calcTotal()
  29. }
  30. func calcTotal() {
  31. let profile = items.map { item -> BasalProfileEntry in
  32. let fotmatter = DateFormatter()
  33. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  34. fotmatter.dateFormat = "HH:mm:ss"
  35. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  36. let minutes = Int(date.timeIntervalSince1970 / 60)
  37. let rate = self.rateValues[item.rateIndex]
  38. return BasalProfileEntry(start: fotmatter.string(from: date), minutes: minutes, rate: rate)
  39. }
  40. var profileWith24hours = profile.map(\.minutes)
  41. profileWith24hours.append(24 * 60)
  42. let pr2 = zip(profile, profileWith24hours.dropFirst())
  43. total = pr2.reduce(0) { $0 + (Decimal($1.1 - $1.0.minutes) / 60) * $1.0.rate }
  44. }
  45. func add() {
  46. var time = 0
  47. var rate = 0
  48. if let last = items.last {
  49. time = last.timeIndex + 1
  50. rate = last.rateIndex
  51. }
  52. let newItem = Item(rateIndex: rate, timeIndex: time)
  53. items.append(newItem)
  54. calcTotal()
  55. }
  56. func save() {
  57. guard hasChanges else { return }
  58. syncInProgress = true
  59. let profile = items.map { item -> BasalProfileEntry in
  60. let formatter = DateFormatter()
  61. formatter.timeZone = TimeZone(secondsFromGMT: 0)
  62. formatter.dateFormat = "HH:mm:ss"
  63. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  64. let minutes = Int(date.timeIntervalSince1970 / 60)
  65. let rate = self.rateValues[item.rateIndex]
  66. return BasalProfileEntry(start: formatter.string(from: date), minutes: minutes, rate: rate)
  67. }
  68. provider.saveProfile(profile)
  69. .receive(on: DispatchQueue.main)
  70. .sink { completion in
  71. self.syncInProgress = false
  72. switch completion {
  73. case .finished:
  74. // Successfully saved and synced
  75. self.initialItems = self.items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  76. Task.detached(priority: .low) {
  77. debug(.nightscout, "Attempting to upload basal rates to Nightscout")
  78. await self.nightscout.uploadProfiles()
  79. }
  80. case .failure:
  81. // Handle the error, show error message
  82. self.showAlert = true
  83. }
  84. } receiveValue: {
  85. // Handle any successful value if needed
  86. print("We were successful")
  87. }
  88. .store(in: &lifetime)
  89. }
  90. func validate() {
  91. DispatchQueue.main.async {
  92. let uniq = Array(Set(self.items))
  93. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  94. sorted.first?.timeIndex = 0
  95. if self.items != sorted {
  96. self.items = sorted
  97. }
  98. }
  99. }
  100. }
  101. }