BasalProfileEditorStateModel.swift 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. import SwiftUI
  2. extension BasalProfileEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Published var syncInProgress = false
  5. @Published var items: [Item] = []
  6. @Published var autotuneProfile: [BasalProfileEntry?] = []
  7. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  8. private(set) var rateValues: [Decimal] = []
  9. var canAdd: Bool {
  10. guard let lastItem = items.last else { return true }
  11. return lastItem.timeIndex < timeValues.count - 1
  12. }
  13. override func subscribe() {
  14. rateValues = provider.supportedBasalRates ?? stride(from: 5.0, to: 1001.0, by: 5.0)
  15. .map { ($0.decimal ?? .zero) / 100 }
  16. items = provider.profile.map { value in
  17. let timeIndex = timeValues.firstIndex(of: Double(value.minutes * 60)) ?? 0
  18. let rateIndex = rateValues.firstIndex(of: value.rate) ?? 0
  19. return Item(rateIndex: rateIndex, timeIndex: timeIndex)
  20. }
  21. var bp: [BasalProfileEntry?] = []
  22. for p in provider.profile {
  23. var np: BasalProfileEntry?
  24. for b in provider.autotune?.basalProfile ?? [] {
  25. if b.start > p.start {
  26. NSLog("Matched \(p) with \(b)")
  27. break
  28. }
  29. np = b
  30. }
  31. bp.append(np)
  32. }
  33. NSLog("basalProfile \(bp)")
  34. autotuneProfile = bp
  35. }
  36. func add() {
  37. var time = 0
  38. var rate = 0
  39. if let last = items.last {
  40. time = last.timeIndex + 1
  41. rate = last.rateIndex
  42. }
  43. let newItem = Item(rateIndex: rate, timeIndex: time)
  44. items.append(newItem)
  45. }
  46. func save() {
  47. syncInProgress = true
  48. let profile = items.map { item -> BasalProfileEntry 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 BasalProfileEntry(start: fotmatter.string(from: date), minutes: minutes, rate: rate)
  56. }
  57. provider.saveProfile(profile)
  58. .receive(on: DispatchQueue.main)
  59. .sink { _ in
  60. self.syncInProgress = false
  61. } receiveValue: {}
  62. .store(in: &lifetime)
  63. }
  64. func read() {
  65. syncInProgress = true
  66. provider.readProfile()
  67. .receive(on: DispatchQueue.main)
  68. .sink { _ in
  69. self.syncInProgress = false
  70. self.subscribe()
  71. } receiveValue: {}
  72. .store(in: &lifetime)
  73. }
  74. func validate() {
  75. DispatchQueue.main.async {
  76. let uniq = Array(Set(self.items))
  77. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  78. sorted.first?.timeIndex = 0
  79. self.items = sorted
  80. }
  81. }
  82. }
  83. }