CarbRatioEditorStateModel.swift 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. import SwiftUI
  2. extension CarbRatioEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Injected() private var nightscout: NightscoutManager!
  5. @Published var items: [Item] = []
  6. @Published var initialItems: [Item] = []
  7. @Published var autotune: Autotune?
  8. @Published var shouldDisplaySaving: Bool = false
  9. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  10. let rateValues = stride(from: 1.0, to: 501.0, by: 1.0).map { ($0.decimal ?? .zero) / 10 }
  11. var canAdd: Bool {
  12. guard let lastItem = items.last else { return true }
  13. return lastItem.timeIndex < timeValues.count - 1
  14. }
  15. var hasChanges: Bool {
  16. if initialItems.count != items.count {
  17. return true
  18. }
  19. for (initialItem, currentItem) in zip(initialItems, items) {
  20. if initialItem.rateIndex != currentItem.rateIndex || initialItem.timeIndex != currentItem.timeIndex {
  21. return true
  22. }
  23. }
  24. return false
  25. }
  26. override func subscribe() {
  27. items = provider.profile.schedule.map { value in
  28. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  29. let rateIndex = rateValues.firstIndex(of: value.ratio) ?? 0
  30. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  31. }
  32. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  33. autotune = provider.autotune
  34. }
  35. func add() {
  36. var time = 0
  37. var rate = 0
  38. if let last = items.last {
  39. time = last.timeIndex + 1
  40. rate = last.rateIndex
  41. }
  42. let newItem = Item(rateIndex: rate, timeIndex: time)
  43. items.append(newItem)
  44. }
  45. func save() {
  46. guard hasChanges else { return }
  47. shouldDisplaySaving = true
  48. let schedule = items.enumerated().map { _, item -> CarbRatioEntry in
  49. let fotmatter = DateFormatter()
  50. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  51. fotmatter.dateFormat = "HH:mm:ss"
  52. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  53. let minutes = Int(date.timeIntervalSince1970 / 60)
  54. let rate = self.rateValues[item.rateIndex]
  55. return CarbRatioEntry(start: fotmatter.string(from: date), offset: minutes, ratio: rate)
  56. }
  57. let profile = CarbRatios(units: .grams, schedule: schedule)
  58. provider.saveProfile(profile)
  59. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  60. Task.detached(priority: .low) {
  61. debug(.nightscout, "Attempting to upload CRs to Nightscout")
  62. await self.nightscout.uploadProfiles()
  63. }
  64. }
  65. func validate() {
  66. DispatchQueue.main.async {
  67. let uniq = Array(Set(self.items))
  68. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  69. sorted.first?.timeIndex = 0
  70. if self.items != sorted {
  71. self.items = sorted
  72. }
  73. }
  74. }
  75. }
  76. }