AddCarbsRootView.swift 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. import CoreData
  2. import SwiftUI
  3. import Swinject
  4. extension AddCarbs {
  5. struct RootView: BaseView {
  6. let resolver: Resolver
  7. @StateObject var state = StateModel()
  8. @State var dish: String = ""
  9. @State var isPromtPresented = false
  10. @State var saved = false
  11. @FetchRequest(
  12. entity: Presets.entity(),
  13. sortDescriptors: [NSSortDescriptor(key: "dish", ascending: true)]
  14. ) var carbPresets: FetchedResults<Presets>
  15. @Environment(\.managedObjectContext) var moc
  16. private var formatter: NumberFormatter {
  17. let formatter = NumberFormatter()
  18. formatter.numberStyle = .decimal
  19. formatter.maximumFractionDigits = 1
  20. return formatter
  21. }
  22. var body: some View {
  23. Form {
  24. if let carbsReq = state.carbsRequired {
  25. Section {
  26. HStack {
  27. Text("Carbs required")
  28. Spacer()
  29. Text(formatter.string(from: carbsReq as NSNumber)! + " g")
  30. }
  31. }
  32. }
  33. Section {
  34. Section {
  35. HStack {
  36. Text("Carbs").fontWeight(.semibold)
  37. Spacer()
  38. DecimalTextField("0", value: $state.carbs, formatter: formatter, autofocus: true, cleanInput: true)
  39. Text("grams").foregroundColor(.secondary)
  40. }.padding(.vertical)
  41. if state.useFPU { proteinAndFat() }
  42. DatePicker("Date", selection: $state.date)
  43. }
  44. }
  45. Section {
  46. Button { state.add() }
  47. label: { Text("Save and continue") }
  48. .disabled(state.carbs <= 0 && state.fat <= 0 && state.protein <= 0)
  49. }
  50. if state.useFPU {
  51. mealPresets
  52. }
  53. }
  54. .onAppear(perform: configureView)
  55. .navigationBarItems(leading: Button("Close", action: state.hideModal))
  56. }
  57. var presetPopover: some View {
  58. Form {
  59. Section(header: Text("Enter Meal Preset Name")) {
  60. TextField("Name Of Dish", text: $dish)
  61. Button {
  62. saved = true
  63. if dish != "", saved {
  64. let preset = Presets(context: moc)
  65. preset.dish = dish
  66. preset.fat = state.fat as NSDecimalNumber
  67. preset.protein = state.protein as NSDecimalNumber
  68. preset.carbs = state.carbs as NSDecimalNumber
  69. try? moc.save()
  70. state.selection = preset
  71. saved = false
  72. isPromtPresented = false
  73. }
  74. }
  75. label: { Text("Save") }
  76. Button {
  77. dish = ""
  78. saved = false
  79. isPromtPresented = false }
  80. label: { Text("Cancel") }
  81. }
  82. }
  83. }
  84. var mealPresets: some View {
  85. Section {
  86. VStack {
  87. Picker("Meal Presets", selection: $state.selection) {
  88. Text("Empty").tag(nil as Presets?)
  89. ForEach(carbPresets, id: \.self) { (preset: Presets) in
  90. Text(preset.dish ?? "").tag(preset as Presets?)
  91. }
  92. }
  93. .pickerStyle(.automatic)
  94. ._onBindingChange($state.selection) { _ in
  95. state.carbs = ((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal
  96. state.fat = ((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal
  97. state.protein = ((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal
  98. }
  99. }
  100. Button {
  101. state.deletePreset()
  102. }
  103. label: { Text("Delete Selected Preset") }
  104. .disabled(state.selection == nil)
  105. .accentColor(.orange)
  106. }
  107. }
  108. @ViewBuilder private func proteinAndFat() -> some View {
  109. HStack {
  110. Text("Protein").foregroundColor(.red).fontWeight(.thin)
  111. Spacer()
  112. DecimalTextField(
  113. "0",
  114. value: $state.protein,
  115. formatter: formatter,
  116. autofocus: false,
  117. cleanInput: true
  118. ).foregroundColor(.loopRed)
  119. Text("grams").foregroundColor(.secondary)
  120. }
  121. HStack {
  122. Text("Fat").foregroundColor(.orange).fontWeight(.thin)
  123. Spacer()
  124. DecimalTextField(
  125. "0",
  126. value: $state.fat,
  127. formatter: formatter,
  128. autofocus: false,
  129. cleanInput: true
  130. )
  131. Text("grams").foregroundColor(.secondary)
  132. }
  133. HStack {
  134. Button {
  135. isPromtPresented = true
  136. }
  137. label: { Text("Save as Preset") }
  138. }
  139. .frame(maxWidth: .infinity, alignment: .trailing)
  140. .disabled(
  141. (state.carbs <= 0 && state.fat <= 0 && state.protein <= 0) ||
  142. (
  143. (((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal) == state
  144. .carbs && (((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal) == state
  145. .fat && (((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal) == state
  146. .protein
  147. )
  148. )
  149. .popover(isPresented: $isPromtPresented) {
  150. presetPopover
  151. }
  152. }
  153. }
  154. }