ISFEditorRootView.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. import SwiftUI
  2. import Swinject
  3. extension ISFEditor {
  4. struct RootView: BaseView {
  5. let resolver: Resolver
  6. @StateObject var state = StateModel()
  7. @State private var editMode = EditMode.inactive
  8. private var dateFormatter: DateFormatter {
  9. let formatter = DateFormatter()
  10. formatter.timeZone = TimeZone(secondsFromGMT: 0)
  11. formatter.timeStyle = .short
  12. return formatter
  13. }
  14. private var rateFormatter: NumberFormatter {
  15. let formatter = NumberFormatter()
  16. formatter.numberStyle = .decimal
  17. formatter.maximumFractionDigits = 2
  18. return formatter
  19. }
  20. var body: some View {
  21. Form {
  22. if let autotune = state.autotune {
  23. Section(header: Text("Autotune")) {
  24. HStack {
  25. Text("Calculated Sensitivity")
  26. Spacer()
  27. if state.units == .mmolL {
  28. Text(rateFormatter.string(from: autotune.sensitivity.asMmolL as NSNumber) ?? "0")
  29. } else {
  30. Text(rateFormatter.string(from: autotune.sensitivity as NSNumber) ?? "0")
  31. }
  32. Text(state.units.rawValue + "/U").foregroundColor(.secondary)
  33. }
  34. }
  35. }
  36. if let newISF = state.autosensISF {
  37. Section(header: Text("Autosens")) {
  38. HStack {
  39. Text("Sensitivity Ratio")
  40. Spacer()
  41. Text(rateFormatter.string(from: state.autosensRatio as NSNumber) ?? "1")
  42. }
  43. HStack {
  44. Text("Calculated Sensitivity")
  45. Spacer()
  46. Text(rateFormatter.string(from: newISF as NSNumber) ?? "0")
  47. Text(state.units.rawValue + "/U").foregroundColor(.secondary)
  48. }
  49. }
  50. }
  51. Section(header: Text("Schedule")) {
  52. list
  53. addButton
  54. }
  55. Section {
  56. Button { state.save() }
  57. label: {
  58. Text("Save")
  59. }
  60. .disabled(state.items.isEmpty)
  61. }
  62. }
  63. .onAppear(perform: configureView)
  64. .navigationTitle("Insulin Sensitivities")
  65. .navigationBarTitleDisplayMode(.automatic)
  66. .navigationBarItems(
  67. trailing: EditButton()
  68. )
  69. .environment(\.editMode, $editMode)
  70. .onAppear {
  71. state.validate()
  72. }
  73. }
  74. private func pickers(for index: Int) -> some View {
  75. GeometryReader { geometry in
  76. VStack {
  77. HStack {
  78. Text("Rate").frame(width: geometry.size.width / 2)
  79. Text("Time").frame(width: geometry.size.width / 2)
  80. }
  81. HStack(spacing: 0) {
  82. Picker(selection: $state.items[index].rateIndex, label: EmptyView()) {
  83. ForEach(0 ..< state.rateValues.count, id: \.self) { i in
  84. Text(
  85. (
  86. self.rateFormatter
  87. .string(from: state.rateValues[i] as NSNumber) ?? ""
  88. ) + " \(state.units.rawValue)/U"
  89. ).tag(i)
  90. }
  91. }
  92. .frame(maxWidth: geometry.size.width / 2)
  93. .clipped()
  94. Picker(selection: $state.items[index].timeIndex, label: EmptyView()) {
  95. ForEach(0 ..< state.timeValues.count, id: \.self) { i in
  96. Text(
  97. self.dateFormatter
  98. .string(from: Date(
  99. timeIntervalSince1970: state
  100. .timeValues[i]
  101. ))
  102. ).tag(i)
  103. }
  104. }
  105. .frame(maxWidth: geometry.size.width / 2)
  106. .clipped()
  107. }
  108. }
  109. }
  110. }
  111. private var list: some View {
  112. List {
  113. ForEach(state.items.indexed(), id: \.1.id) { index, item in
  114. NavigationLink(destination: pickers(for: index)) {
  115. HStack {
  116. Text("Rate").foregroundColor(.secondary)
  117. Text(
  118. "\(rateFormatter.string(from: state.rateValues[item.rateIndex] as NSNumber) ?? "0") \(state.units.rawValue)/U"
  119. )
  120. Spacer()
  121. Text("starts at").foregroundColor(.secondary)
  122. Text(
  123. "\(dateFormatter.string(from: Date(timeIntervalSince1970: state.timeValues[item.timeIndex])))"
  124. )
  125. }
  126. }
  127. .moveDisabled(true)
  128. }
  129. .onDelete(perform: onDelete)
  130. }
  131. }
  132. private var addButton: some View {
  133. guard state.canAdd else {
  134. return AnyView(EmptyView())
  135. }
  136. switch editMode {
  137. case .inactive:
  138. return AnyView(Button(action: onAdd) { Text("Add") })
  139. default:
  140. return AnyView(EmptyView())
  141. }
  142. }
  143. func onAdd() {
  144. state.add()
  145. }
  146. private func onDelete(offsets: IndexSet) {
  147. state.items.remove(atOffsets: offsets)
  148. state.validate()
  149. }
  150. }
  151. }