CarbRatioEditorStateModel.swift 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. import SwiftUI
  2. extension CarbRatioEditor {
  3. final class StateModel: BaseStateModel<Provider> {
  4. @Published var items: [Item] = []
  5. @Published var initialItems: [Item] = []
  6. @Published var autotune: Autotune?
  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: 1.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. autotune = provider.autotune
  33. }
  34. func add() {
  35. var time = 0
  36. var rate = 0
  37. if let last = items.last {
  38. time = last.timeIndex + 1
  39. rate = last.rateIndex
  40. }
  41. let newItem = Item(rateIndex: rate, timeIndex: time)
  42. items.append(newItem)
  43. }
  44. func save() {
  45. guard hasChanges else { return }
  46. shouldDisplaySaving = true
  47. let schedule = items.enumerated().map { _, item -> CarbRatioEntry in
  48. let fotmatter = DateFormatter()
  49. fotmatter.timeZone = TimeZone(secondsFromGMT: 0)
  50. fotmatter.dateFormat = "HH:mm:ss"
  51. let date = Date(timeIntervalSince1970: self.timeValues[item.timeIndex])
  52. let minutes = Int(date.timeIntervalSince1970 / 60)
  53. let rate = self.rateValues[item.rateIndex]
  54. return CarbRatioEntry(start: fotmatter.string(from: date), offset: minutes, ratio: rate)
  55. }
  56. let profile = CarbRatios(units: .grams, schedule: schedule)
  57. provider.saveProfile(profile)
  58. initialItems = items.map { Item(rateIndex: $0.rateIndex, timeIndex: $0.timeIndex) }
  59. }
  60. func validate() {
  61. DispatchQueue.main.async {
  62. let uniq = Array(Set(self.items))
  63. let sorted = uniq.sorted { $0.timeIndex < $1.timeIndex }
  64. sorted.first?.timeIndex = 0
  65. if self.items != sorted {
  66. self.items = sorted
  67. }
  68. }
  69. }
  70. }
  71. }