TargetsEditorStateModel.swift 3.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. import SwiftUI
  2. extension TargetsEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Published var items: [Item] = []
  5. @Published var initialItems: [Item] = []
  6. @Published var shouldDisplaySaving: Bool = false
  7. let timeValues = stride(from: 0.0, to: 1.days.timeInterval, by: 30.minutes.timeInterval).map { $0 }
  8. var rateValues: [Decimal] {
  9. var values = stride(from: 72.0, to: 180.01, by: 1.0).map { Decimal($0) }
  10. if units == .mmolL {
  11. values = values.filter { Int(truncating: $0 as NSNumber) % 2 == 0 }
  12. }
  13. return values
  14. }
  15. var canAdd: Bool {
  16. guard let lastItem = items.last else { return true }
  17. return lastItem.timeIndex < timeValues.count - 1
  18. }
  19. var hasChanges: Bool {
  20. initialItems != items
  21. }
  22. private(set) var units: GlucoseUnits = .mgdL
  23. override func subscribe() {
  24. units = settingsManager.settings.units
  25. let profile = provider.profile
  26. items = profile.targets.map { value in
  27. let timeIndex = timeValues.firstIndex(of: Double(value.offset * 60)) ?? 0
  28. let lowIndex = rateValues.firstIndex(of: value.low) ?? 0
  29. let highIndex = rateValues.firstIndex(of: value.high) ?? 0
  30. return Item(lowIndex: lowIndex, highIndex: highIndex, timeIndex: timeIndex)
  31. }
  32. initialItems = items.map { Item(lowIndex: $0.lowIndex, highIndex: $0.highIndex, timeIndex: $0.timeIndex) }
  33. }
  34. func add() {
  35. var time = 0
  36. var low = 0
  37. var high = 0
  38. if let last = items.last {
  39. time = last.timeIndex + 1
  40. low = last.lowIndex
  41. high = low
  42. }
  43. let newItem = Item(lowIndex: low, highIndex: high, timeIndex: time)
  44. items.append(newItem)
  45. }
  46. func save() {
  47. guard hasChanges else { return }
  48. shouldDisplaySaving.toggle()
  49. let targets = items.map { item -> BGTargetEntry in
  50. let formatter = DateFormatter()
  51. formatter.timeZone = TimeZone(secondsFromGMT: 0)
  52. formatter.dateFormat = "HH:mm:ss"
  53. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  54. let minutes = Int(date.timeIntervalSince1970 / 60)
  55. let low = self.rateValues[item.lowIndex]
  56. let high = low
  57. return BGTargetEntry(low: low, high: high, start: formatter.string(from: date), offset: minutes)
  58. }
  59. let profile = BGTargets(units: .mgdL, userPreferredUnits: .mgdL, targets: targets)
  60. provider.saveProfile(profile)
  61. initialItems = items.map { Item(lowIndex: $0.lowIndex, highIndex: $0.highIndex, timeIndex: $0.timeIndex) }
  62. }
  63. func validate() {
  64. DispatchQueue.main.async {
  65. let uniq = Array(Set(self.items))
  66. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  67. .map { item -> Item in
  68. Item(lowIndex: item.lowIndex, highIndex: item.highIndex, timeIndex: item.timeIndex)
  69. }
  70. sorted.first?.timeIndex = 0
  71. self.items = sorted
  72. if self.items.isEmpty {
  73. self.units = self.settingsManager.settings.units
  74. }
  75. }
  76. }
  77. }
  78. }