| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624 |
- import Charts
- import CoreData
- import LoopKitUI
- import SwiftUI
- import Swinject
- extension Bolus {
- struct RootView: BaseView {
- enum FocusedField {
- case carbs
- case fat
- case protein
- }
- @FocusState private var focusedField: FocusedField?
- let resolver: Resolver
- @StateObject var state = StateModel()
- @State private var showAlert = false
- @State private var autofocus: Bool = true
- @State private var calculatorDetent = PresentationDetent.medium
- @State private var pushed: Bool = false
- @State private var isPromptPresented: Bool = false
- @State private var dish: String = ""
- @State private var saved: Bool = false
- @State private var debounce: DispatchWorkItem?
- @Environment(\.managedObjectContext) var moc
- private enum Config {
- static let dividerHeight: CGFloat = 2
- static let spacing: CGFloat = 3
- }
- @Environment(\.colorScheme) var colorScheme
- @FetchRequest(
- entity: MealPresetStored.entity(),
- sortDescriptors: [NSSortDescriptor(key: "dish", ascending: true)]
- ) var carbPresets: FetchedResults<MealPresetStored>
- private var formatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- formatter.maximumFractionDigits = 2
- return formatter
- }
- private var mealFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- formatter.maximumFractionDigits = 1
- return formatter
- }
- private var gluoseFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- if state.units == .mmolL {
- formatter.maximumFractionDigits = 1
- } else { formatter.maximumFractionDigits = 0 }
- return formatter
- }
- private var fractionDigits: Int {
- if state.units == .mmolL {
- return 1
- } else { return 0 }
- }
- private var color: LinearGradient {
- colorScheme == .dark ? LinearGradient(
- gradient: Gradient(colors: [
- Color.bgDarkBlue,
- Color.bgDarkerDarkBlue
- ]),
- startPoint: .top,
- endPoint: .bottom
- )
- :
- LinearGradient(
- gradient: Gradient(colors: [Color.gray.opacity(0.1)]),
- startPoint: .top,
- endPoint: .bottom
- )
- }
- private var empty: Bool {
- state.useFPUconversion ? (state.carbs <= 0 && state.fat <= 0 && state.protein <= 0) : (state.carbs <= 0)
- }
- /// Handles macro input (carb, fat, protein) in a debounced fashion.
- func handleDebouncedInput() {
- debounce?.cancel()
- debounce = DispatchWorkItem { [self] in
- state.insulinCalculated = state.calculateInsulin()
- }
- if let debounce = debounce {
- DispatchQueue.main.asyncAfter(deadline: .now() + 0.35, execute: debounce)
- }
- }
- private var presetPopover: some View {
- Form {
- Section {
- TextField("Name Of Dish", text: $dish)
- Button {
- saved = true
- if dish != "", saved {
- let preset = MealPresetStored(context: moc)
- preset.dish = dish
- preset.fat = state.fat as NSDecimalNumber
- preset.protein = state.protein as NSDecimalNumber
- preset.carbs = state.carbs as NSDecimalNumber
- if self.moc.hasChanges {
- try? moc.save()
- }
- state.addNewPresetToWaitersNotepad(dish)
- saved = false
- isPromptPresented = false
- }
- }
- label: { Text("Save") }
- Button {
- dish = ""
- saved = false
- isPromptPresented = false }
- label: { Text("Cancel") }
- } header: { Text("Enter Meal Preset Name") }
- }
- }
- private var minusButton: some View {
- Button {
- if state.carbs != 0,
- (state.carbs - (((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal) as Decimal) >= 0
- {
- state.carbs -= (((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal)
- } else { state.carbs = 0 }
- if state.fat != 0,
- (state.fat - (((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal) as Decimal) >= 0
- {
- state.fat -= (((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal)
- } else { state.fat = 0 }
- if state.protein != 0,
- (state.protein - (((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal) as Decimal) >= 0
- {
- state.protein -= (((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal)
- } else { state.protein = 0 }
- state.removePresetFromNewMeal()
- if state.carbs == 0, state.fat == 0, state.protein == 0 { state.summation = [] }
- }
- label: { Image(systemName: "minus.circle.fill")
- .font(.system(size: 20))
- }
- .disabled(
- state
- .selection == nil ||
- (
- !state.summation
- .contains(state.selection?.dish ?? "") && (state.selection?.dish ?? "") != ""
- )
- )
- .buttonStyle(.borderless)
- .tint(.blue)
- }
- private var plusButton: some View {
- Button {
- state.carbs += ((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal
- state.fat += ((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal
- state.protein += ((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal
- state.addPresetToNewMeal()
- }
- label: { Image(systemName: "plus.circle.fill")
- .font(.system(size: 20))
- }
- .disabled(state.selection == nil)
- .buttonStyle(.borderless)
- .tint(.blue)
- }
- private var mealPresets: some View {
- Section {
- HStack {
- if state.selection != nil {
- minusButton
- }
- Picker("Preset", selection: $state.selection) {
- Text("Saved Food").tag(nil as MealPresetStored?)
- ForEach(carbPresets, id: \.self) { (preset: MealPresetStored) in
- Text(preset.dish ?? "").tag(preset as MealPresetStored?)
- }
- }
- .labelsHidden()
- .frame(maxWidth: .infinity, alignment: .center)
- ._onBindingChange($state.selection) { _ in
- state.carbs += ((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal
- state.fat += ((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal
- state.protein += ((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal
- state.addToSummation()
- }
- if state.selection != nil {
- plusButton
- }
- }
- HStack {
- Button("Delete Preset") {
- showAlert.toggle()
- }
- .disabled(state.selection == nil)
- .tint(.orange)
- .buttonStyle(.borderless)
- .alert(
- "Delete preset '\(state.selection?.dish ?? "")'?",
- isPresented: $showAlert,
- actions: {
- Button("No", role: .cancel) {}
- Button("Yes", role: .destructive) {
- state.deletePreset()
- state.carbs += ((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal
- state.fat += ((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal
- state.protein += ((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal
- state.addPresetToNewMeal()
- }
- }
- )
- Spacer()
- Button {
- isPromptPresented = true
- }
- label: { Text("Save as Preset") }
- .buttonStyle(.borderless)
- .disabled(
- empty ||
- (
- (((state.selection?.carbs ?? 0) as NSDecimalNumber) as Decimal) == state
- .carbs && (((state.selection?.fat ?? 0) as NSDecimalNumber) as Decimal) == state
- .fat && (((state.selection?.protein ?? 0) as NSDecimalNumber) as Decimal) == state
- .protein
- )
- )
- }
- }
- }
- @ViewBuilder private func proteinAndFat() -> some View {
- HStack {
- Text("Fat").foregroundColor(.orange)
- Spacer()
- TextFieldWithToolBar(text: $state.fat, placeholder: "0", keyboardType: .numberPad, numberFormatter: mealFormatter)
- Text("g").foregroundColor(.secondary)
- }
- HStack {
- Text("Protein").foregroundColor(.red)
- Spacer()
- TextFieldWithToolBar(
- text: $state.protein,
- placeholder: "0",
- keyboardType: .numberPad,
- numberFormatter: mealFormatter
- )
- Text("g").foregroundColor(.secondary)
- }
- }
- @ViewBuilder private func carbsTextField() -> some View {
- HStack {
- Text("Carbs").fontWeight(.semibold)
- Spacer()
- TextFieldWithToolBar(
- text: $state.carbs,
- placeholder: "0",
- keyboardType: .numberPad,
- numberFormatter: mealFormatter
- )
- .onChange(of: state.carbs) { _ in
- if state.carbs > 0 {
- handleDebouncedInput()
- }
- }
- Text("g").foregroundColor(.secondary)
- }
- }
- var body: some View {
- ZStack(alignment: .center) {
- VStack {
- Form {
- Section {
- carbsTextField()
- if state.useFPUconversion {
- proteinAndFat()
- }
- // Summary when combining presets
- if state.waitersNotepad() != "" {
- HStack {
- Text("Total")
- let test = state.waitersNotepad().components(separatedBy: ", ").removeDublicates()
- HStack(spacing: 0) {
- ForEach(test, id: \.self) {
- Text($0).foregroundStyle(Color.blue).font(.footnote)
- Text($0 == test[test.count - 1] ? "" : ", ")
- }
- }.frame(maxWidth: .infinity, alignment: .trailing)
- }
- }
- // Time
- HStack {
- Text("Time").foregroundStyle(Color.secondary)
- Spacer()
- if !pushed {
- Button {
- pushed = true
- } label: { Text("Now") }.buttonStyle(.borderless).foregroundColor(.secondary)
- .padding(.trailing, 5)
- } else {
- Button { state.date = state.date.addingTimeInterval(-15.minutes.timeInterval) }
- label: { Image(systemName: "minus.circle") }.tint(.blue).buttonStyle(.borderless)
- DatePicker(
- "Time",
- selection: $state.date,
- displayedComponents: [.hourAndMinute]
- ).controlSize(.mini)
- .labelsHidden()
- Button {
- state.date = state.date.addingTimeInterval(15.minutes.timeInterval)
- }
- label: { Image(systemName: "plus.circle") }.tint(.blue).buttonStyle(.borderless)
- }
- }
- .popover(isPresented: $isPromptPresented) {
- presetPopover
- }
- }.listRowBackground(Color.chart)
- if state.displayPresets {
- Section {
- mealPresets
- }.listRowBackground(Color.chart)
- }
- Section {
- HStack {
- Button(action: {
- state.showInfo.toggle()
- }, label: {
- Image(systemName: "info.circle")
- Text("Calculations")
- })
- .foregroundStyle(.blue)
- .font(.footnote)
- .buttonStyle(PlainButtonStyle())
- .frame(maxWidth: .infinity, alignment: .leading)
- if state.fattyMeals {
- Spacer()
- Toggle(isOn: $state.useFattyMealCorrectionFactor) {
- Text("Fatty Meal")
- }
- .toggleStyle(CheckboxToggleStyle())
- .font(.footnote)
- .onChange(of: state.useFattyMealCorrectionFactor) { _ in
- state.insulinCalculated = state.calculateInsulin()
- if state.useFattyMealCorrectionFactor {
- state.useSuperBolus = false
- }
- }
- }
- if state.sweetMeals {
- Spacer()
- Toggle(isOn: $state.useSuperBolus) {
- Text("Super Bolus")
- }
- .toggleStyle(CheckboxToggleStyle())
- .font(.footnote)
- .onChange(of: state.useSuperBolus) { _ in
- state.insulinCalculated = state.calculateInsulin()
- if state.useSuperBolus {
- state.useFattyMealCorrectionFactor = false
- }
- }
- }
- }
- HStack {
- Text("Recommended Bolus")
- Spacer()
- Text(
- formatter
- .string(from: Double(state.insulinCalculated) as NSNumber) ?? ""
- )
- Text(
- NSLocalizedString(
- " U",
- comment: "Unit in number of units delivered (keep the space character!)"
- )
- ).foregroundColor(.secondary)
- }.contentShape(Rectangle())
- .onTapGesture { state.amount = state.insulinCalculated }
- HStack {
- Text("Bolus")
- Spacer()
- TextFieldWithToolBar(
- text: $state.amount,
- placeholder: "0",
- textColor: colorScheme == .dark ? .white : .blue,
- maxLength: 5,
- numberFormatter: formatter
- )
- Text(" U").foregroundColor(.secondary)
- }
- if state.amount > 0 {
- HStack {
- Text("External insulin")
- Spacer()
- Toggle("", isOn: $state.externalInsulin).toggleStyle(Checkbox())
- }
- }
- }.listRowBackground(Color.chart)
- Section {
- ForeCastChart(state: state, units: $state.units)
- .padding(.vertical)
- }.listRowBackground(Color.chart)
- }
- }
- .safeAreaInset(edge: .bottom, spacing: 0) {
- stickyButton
- }.blur(radius: state.waitForSuggestion ? 5 : 0)
- if state.waitForSuggestion {
- CustomProgressView(text: progressText.rawValue)
- }
- }
- .scrollContentBackground(.hidden).background(color)
- .blur(radius: state.showInfo ? 3 : 0)
- .navigationTitle("Treatments")
- .navigationBarTitleDisplayMode(.inline)
- .toolbar(content: {
- ToolbarItem(placement: .topBarLeading) {
- Button {
- state.hideModal()
- } label: {
- Text("Close")
- }
- }
- })
- .onAppear {
- configureView {
- state.insulinCalculated = state.calculateInsulin()
- }
- }
- .onDisappear {
- state.addButtonPressed = false
- }
- .sheet(isPresented: $state.showInfo) {
- PopupView(state: state)
- .presentationDetents(
- [.fraction(0.9), .large],
- selection: $calculatorDetent
- )
- }
- }
- var progressText: ProgressText {
- switch (state.amount > 0, state.carbs > 0) {
- case (true, true):
- return .updatingIOBandCOB
- case (false, true):
- return .updatingCOB
- case (true, false):
- return .updatingIOB
- default:
- return .updatingTreatments
- }
- }
- var stickyButton: some View {
- ZStack {
- Rectangle()
- .frame(width: UIScreen.main.bounds.width, height: 120).offset(y: 40)
- .shadow(
- color: colorScheme == .dark ? Color(red: 0.02745098039, green: 0.1098039216, blue: 0.1411764706) :
- Color.black.opacity(0.33),
- radius: 3
- )
- .foregroundStyle(Color.chart)
- Button {
- state.invokeTreatmentsTask()
- } label: {
- taskButtonLabel
- .font(.headline)
- .foregroundStyle(Color.white)
- .frame(maxWidth: .infinity, alignment: .center)
- .frame(minHeight: 50)
- }
- .disabled(disableTaskButton)
- .background(
- (state.externalInsulin ? externalBolusLimit : pumpBolusLimit) ? Color(.systemRed) :
- Color(.systemBlue)
- )
- .shadow(radius: 3)
- .clipShape(RoundedRectangle(cornerRadius: 8))
- .padding()
- .offset(y: 20)
- }
- }
- private var taskButtonLabel: some View {
- let hasInsulin = state.amount > 0
- let hasCarbs = state.carbs > 0
- let hasFatOrProtein = state.fat > 0 || state.protein > 0
- switch (hasInsulin, hasCarbs, hasFatOrProtein) {
- case (true, true, true):
- return Text(
- state
- .externalInsulin ? (
- externalBolusLimit ? "Manual bolus exceeds max bolus!" : "Log meal and external insulin"
- ) :
- (pumpBolusLimit ? "Pump bolus exceeds max bolus!" : "Log meal and enact bolus")
- )
- case (true, true, false):
- return Text(
- state
- .externalInsulin ?
- (externalBolusLimit ? "Manual bolus exceeds max bolus!" : "Log carbs and external insulin") :
- (pumpBolusLimit ? "Pump bolus exceeds max bolus!" : "Log carbs and enact bolus")
- )
- case (true, false, true):
- return Text(
- state
- .externalInsulin ?
- (externalBolusLimit ? "Manual bolus exceeds max bolus!" : "Log FPUs and external insulin") :
- (pumpBolusLimit ? "Pump bolus exceeds max bolus!" : "Log FPUs and enact bolus")
- )
- case (true, false, false):
- return Text(
- state
- .externalInsulin ? (externalBolusLimit ? "Manual bolus exceeds max bolus!" : "Log external insulin") :
- (pumpBolusLimit ? "Pump bolus exceeds max bolus!" : "Enact bolus")
- )
- case (false, true, true):
- return Text("Log meal")
- case (false, true, false):
- return Text("Log carbs")
- case (false, false, true):
- return Text("Log FPUs")
- default:
- return Text("Continue without treatment")
- }
- }
- private var pumpBolusLimit: Bool {
- state.amount > state.maxBolus
- }
- private var externalBolusLimit: Bool {
- state.amount > state.maxBolus * 3
- }
- private var disableTaskButton: Bool {
- state.amount > 0 ? (state.externalInsulin ? externalBolusLimit : pumpBolusLimit) : false
- }
- }
- struct DividerDouble: View {
- var body: some View {
- VStack(spacing: 2) {
- Rectangle()
- .frame(height: 1)
- .foregroundColor(.gray.opacity(0.65))
- Rectangle()
- .frame(height: 1)
- .foregroundColor(.gray.opacity(0.65))
- }
- .frame(height: 4)
- .padding(.vertical)
- }
- }
- struct DividerCustom: View {
- var body: some View {
- Rectangle()
- .frame(height: 1)
- .foregroundColor(.gray.opacity(0.65))
- .padding(.vertical)
- }
- }
- }
- // fix iOS 15 bug
- struct ActivityIndicator: UIViewRepresentable {
- @Binding var isAnimating: Bool
- let style: UIActivityIndicatorView.Style
- func makeUIView(context _: UIViewRepresentableContext<ActivityIndicator>) -> UIActivityIndicatorView {
- UIActivityIndicatorView(style: style)
- }
- func updateUIView(_ uiView: UIActivityIndicatorView, context _: UIViewRepresentableContext<ActivityIndicator>) {
- isAnimating ? uiView.startAnimating() : uiView.stopAnimating()
- }
- }
|