CarbRatioEditorStateModel.swift 3.4 KB

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