| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726 |
- import CoreData
- import SpriteKit
- import SwiftDate
- import SwiftUI
- import Swinject
- extension Home {
- struct RootView: BaseView {
- let resolver: Resolver
- @StateObject var state = StateModel()
- @State var isStatusPopupPresented = false
- @State var showCancelAlert = false
- @Environment(\.managedObjectContext) var moc
- @Environment(\.colorScheme) var colorScheme
- @FetchRequest(
- entity: Override.entity(),
- sortDescriptors: [NSSortDescriptor(key: "date", ascending: false)]
- ) var fetchedPercent: FetchedResults<Override>
- @FetchRequest(
- entity: OverridePresets.entity(),
- sortDescriptors: [NSSortDescriptor(key: "name", ascending: true)], predicate: NSPredicate(
- format: "name != %@", "" as String
- )
- ) var fetchedProfiles: FetchedResults<OverridePresets>
- @FetchRequest(
- entity: TempTargets.entity(),
- sortDescriptors: [NSSortDescriptor(key: "date", ascending: false)]
- ) var sliderTTpresets: FetchedResults<TempTargets>
- @FetchRequest(
- entity: TempTargetsSlider.entity(),
- sortDescriptors: [NSSortDescriptor(key: "date", ascending: false)]
- ) var enactedSliderTT: FetchedResults<TempTargetsSlider>
- private var numberFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- formatter.maximumFractionDigits = 2
- return formatter
- }
- private var fetchedTargetFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- if state.units == .mmolL {
- formatter.maximumFractionDigits = 1
- } else { formatter.maximumFractionDigits = 0 }
- return formatter
- }
- private var targetFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- formatter.maximumFractionDigits = 1
- return formatter
- }
- private var tirFormatter: NumberFormatter {
- let formatter = NumberFormatter()
- formatter.numberStyle = .decimal
- formatter.maximumFractionDigits = 0
- return formatter
- }
- private var dateFormatter: DateFormatter {
- let dateFormatter = DateFormatter()
- dateFormatter.timeStyle = .short
- return dateFormatter
- }
- private var spriteScene: SKScene {
- let scene = SnowScene()
- scene.scaleMode = .resizeFill
- scene.backgroundColor = .clear
- return scene
- }
- @ViewBuilder func header(_: GeometryProxy) -> some View {
- HStack {
- Spacer()
- pumpView
- Spacer()
- }
- }
- var cobIobView: some View {
- VStack(alignment: .leading, spacing: 12) {
- HStack {
- Text("IOB").font(.footnote).foregroundColor(.secondary)
- Text(
- (numberFormatter.string(from: (state.suggestion?.iob ?? 0) as NSNumber) ?? "0") +
- NSLocalizedString(" U", comment: "Insulin unit")
- )
- .font(.footnote).fontWeight(.bold)
- }.frame(alignment: .top)
- HStack {
- Text("COB").font(.footnote).foregroundColor(.secondary)
- Text(
- (numberFormatter.string(from: (state.suggestion?.cob ?? 0) as NSNumber) ?? "0") +
- NSLocalizedString(" g", comment: "gram of carbs")
- )
- .font(.footnote).fontWeight(.bold)
- }.frame(alignment: .bottom)
- }
- }
- var cobIobView2: some View {
- HStack {
- Text("IOB").font(.callout).foregroundColor(.secondary)
- Text(
- (numberFormatter.string(from: (state.suggestion?.iob ?? 0) as NSNumber) ?? "0") +
- NSLocalizedString(" U", comment: "Insulin unit")
- )
- .font(.callout).fontWeight(.bold)
- Spacer()
- Text("COB").font(.callout).foregroundColor(.secondary)
- Text(
- (numberFormatter.string(from: (state.suggestion?.cob ?? 0) as NSNumber) ?? "0") +
- NSLocalizedString(" g", comment: "gram of carbs")
- )
- .font(.callout).fontWeight(.bold)
- Spacer()
- }
- }
- var glucoseView: some View {
- CurrentGlucoseView(
- recentGlucose: $state.recentGlucose,
- timerDate: $state.timerDate,
- delta: $state.glucoseDelta,
- units: $state.units,
- alarm: $state.alarm,
- lowGlucose: $state.lowGlucose,
- highGlucose: $state.highGlucose
- )
- .onTapGesture {
- if state.alarm == nil {
- state.openCGM()
- } else {
- state.showModal(for: .snooze)
- }
- }
- .onLongPressGesture {
- let impactHeavy = UIImpactFeedbackGenerator(style: .heavy)
- impactHeavy.impactOccurred()
- if state.alarm == nil {
- state.showModal(for: .snooze)
- } else {
- state.openCGM()
- }
- }
- }
- var pumpView: some View {
- PumpView(
- reservoir: $state.reservoir,
- battery: $state.battery,
- name: $state.pumpName,
- expiresAtDate: $state.pumpExpiresAtDate,
- timerDate: $state.timerDate,
- state: state
- )
- .onTapGesture {
- if state.pumpDisplayState != nil {
- state.setupPump = true
- }
- }
- }
- var loopView: some View {
- LoopView(
- suggestion: $state.suggestion,
- enactedSuggestion: $state.enactedSuggestion,
- closedLoop: $state.closedLoop,
- timerDate: $state.timerDate,
- isLooping: $state.isLooping,
- lastLoopDate: $state.lastLoopDate,
- manualTempBasal: $state.manualTempBasal
- ).onTapGesture {
- isStatusPopupPresented = true
- }.onLongPressGesture {
- let impactHeavy = UIImpactFeedbackGenerator(style: .heavy)
- impactHeavy.impactOccurred()
- state.runLoop()
- }
- }
- var tempBasalString: String? {
- guard let tempRate = state.tempRate else {
- return nil
- }
- let rateString = numberFormatter.string(from: tempRate as NSNumber) ?? "0"
- var manualBasalString = ""
- if state.apsManager.isManualTempBasal {
- manualBasalString = NSLocalizedString(
- " - Manual Basal ⚠️",
- comment: "Manual Temp basal"
- )
- }
- return rateString + NSLocalizedString(" U/hr", comment: "Unit per hour with space") + manualBasalString
- }
- var tempTargetString: String? {
- guard let tempTarget = state.tempTarget else {
- return nil
- }
- let target = tempTarget.targetBottom ?? 0
- let unitString = targetFormatter.string(from: (tempTarget.targetBottom?.asMmolL ?? 0) as NSNumber) ?? ""
- let rawString = (tirFormatter.string(from: (tempTarget.targetBottom ?? 0) as NSNumber) ?? "") + " " + state.units
- .rawValue
- var string = ""
- if sliderTTpresets.first?.active ?? false {
- let hbt = sliderTTpresets.first?.hbt ?? 0
- string = ", " + (tirFormatter.string(from: state.infoPanelTTPercentage(hbt, target) as NSNumber) ?? "") + " %"
- }
- let percentString = state
- .units == .mmolL ? (unitString + " mmol/L" + string) : (rawString + (string == "0" ? "" : string))
- return tempTarget.displayName + " " + percentString
- }
- var overrideString: String? {
- guard fetchedPercent.first?.enabled ?? false else {
- return nil
- }
- var percentString = "\((fetchedPercent.first?.percentage ?? 100).formatted(.number)) %"
- var target = (fetchedPercent.first?.target ?? 100) as Decimal
- let indefinite = (fetchedPercent.first?.indefinite ?? false)
- let unit = state.units.rawValue
- if state.units == .mmolL {
- target = target.asMmolL
- }
- var targetString = (fetchedTargetFormatter.string(from: target as NSNumber) ?? "") + " " + unit
- if tempTargetString != nil || target == 0 { targetString = "" }
- percentString = percentString == "100 %" ? "" : percentString
- let duration = (fetchedPercent.first?.duration ?? 0) as Decimal
- let addedMinutes = Int(duration)
- let date = fetchedPercent.first?.date ?? Date()
- var newDuration: Decimal = 0
- if date.addingTimeInterval(addedMinutes.minutes.timeInterval) > Date() {
- newDuration = Decimal(Date().distance(to: date.addingTimeInterval(addedMinutes.minutes.timeInterval)).minutes)
- }
- var durationString = indefinite ?
- "" : newDuration >= 1 ?
- (newDuration.formatted(.number.grouping(.never).rounded().precision(.fractionLength(0))) + " min") :
- (
- newDuration > 0 ? (
- (newDuration * 60).formatted(.number.grouping(.never).rounded().precision(.fractionLength(0))) + " s"
- ) :
- ""
- )
- let smbToggleString = (fetchedPercent.first?.smbIsOff ?? false) ? " \u{20e0}" : ""
- var comma1 = ", "
- var comma2 = comma1
- var comma3 = comma1
- if targetString == "" || percentString == "" { comma1 = "" }
- if durationString == "" { comma2 = "" }
- if smbToggleString == "" { comma3 = "" }
- if percentString == "", targetString == "" {
- comma1 = ""
- comma2 = ""
- }
- if percentString == "", targetString == "", smbToggleString == "" {
- durationString = ""
- comma1 = ""
- comma2 = ""
- comma3 = ""
- }
- if durationString == "" {
- comma2 = ""
- }
- if smbToggleString == "" {
- comma3 = ""
- }
- if durationString == "", !indefinite {
- return nil
- }
- return percentString + comma1 + targetString + comma2 + durationString + comma3 + smbToggleString
- }
- var infoPanel: some View {
- HStack(alignment: .center) {
- if state.pumpSuspended {
- Text("Pump suspended")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.loopGray)
- .padding(.leading, 8)
- } else if let tempBasalString = tempBasalString {
- Text(tempBasalString)
- .font(.system(size: 12, weight: .bold))
- .foregroundColor(.insulin)
- .padding(.leading, 8)
- }
- if let tempTargetString = tempTargetString {
- Text(tempTargetString)
- .font(.caption)
- .foregroundColor(.secondary)
- }
- Spacer()
- if let overrideString = overrideString {
- Text("👤 " + overrideString)
- .font(.system(size: 12))
- .foregroundColor(.secondary)
- .padding(.trailing, 8)
- }
- if state.closedLoop, state.settingsManager.preferences.maxIOB == 0 {
- Text("Max IOB: 0").font(.callout).foregroundColor(.orange).padding(.trailing, 20)
- }
- if let progress = state.bolusProgress {
- Text("Bolusing")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.insulin)
- ProgressView(value: Double(progress))
- .progressViewStyle(BolusProgressViewStyle())
- .padding(.trailing, 8)
- .onTapGesture {
- state.cancelBolus()
- }
- }
- }
- .frame(maxWidth: .infinity, maxHeight: 30)
- }
- var legendPanel: some View {
- ZStack {
- HStack(alignment: .center) {
- Group {
- Circle().fill(Color.loopGreen).frame(width: 8, height: 8)
- Text("BG")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.loopGreen)
- }
- Group {
- Circle().fill(Color.insulin).frame(width: 8, height: 8)
- .padding(.leading, 8)
- Text("IOB")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.insulin)
- }
- Group {
- Circle().fill(Color.zt).frame(width: 8, height: 8)
- .padding(.leading, 8)
- Text("ZT")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.zt)
- }
- Group {
- Circle().fill(Color.loopYellow).frame(width: 8, height: 8)
- .padding(.leading, 8)
- Text("COB")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.loopYellow)
- }
- Group {
- Circle().fill(Color.uam).frame(width: 8, height: 8)
- .padding(.leading, 8)
- Text("UAM")
- .font(.system(size: 12, weight: .bold)).foregroundColor(.uam)
- }
- if let eventualBG = state.eventualBG {
- Text(
- "⇢ " + numberFormatter.string(
- from: (state.units == .mmolL ? eventualBG.asMmolL : Decimal(eventualBG)) as NSNumber
- )!
- )
- .font(.system(size: 12, weight: .bold)).foregroundColor(.secondary)
- }
- }
- .frame(maxWidth: .infinity)
- .padding([.bottom], 20)
- }
- }
- var mainChart: some View {
- ZStack {
- if state.animatedBackground {
- SpriteView(scene: spriteScene, options: [.allowsTransparency])
- .ignoresSafeArea()
- .frame(minWidth: 0, maxWidth: .infinity, minHeight: 0, maxHeight: .infinity)
- }
- MainChartView(
- glucose: $state.glucose,
- isManual: $state.isManual,
- suggestion: $state.suggestion,
- tempBasals: $state.tempBasals,
- boluses: $state.boluses,
- suspensions: $state.suspensions,
- announcement: $state.announcement,
- hours: .constant(state.filteredHours),
- maxBasal: $state.maxBasal,
- autotunedBasalProfile: $state.autotunedBasalProfile,
- basalProfile: $state.basalProfile,
- tempTargets: $state.tempTargets,
- carbs: $state.carbs,
- timerDate: $state.timerDate,
- units: $state.units,
- smooth: $state.smooth,
- highGlucose: $state.highGlucose,
- lowGlucose: $state.lowGlucose,
- screenHours: $state.screenHours,
- displayXgridLines: $state.displayXgridLines,
- displayYgridLines: $state.displayYgridLines,
- thresholdLines: $state.thresholdLines,
- state: state
- )
- }
- .padding(.bottom)
- .modal(for: .dataTable, from: self)
- }
- // MARK: PICKER IN SEGEMENTED STYLE TO CHOOSE THE X AXIS SCALE OF THE GRAPH
- @ViewBuilder private func pickerPanel(_: GeometryProxy) -> some View {
- HStack {
- Picker("Scale", selection: $state.scale) {
- ForEach(Home.StateModel.Scale.allCases) { scale in
- Text("\(scale.rawValue)h").tag(Optional(scale))
- }
- }
- .pickerStyle(.segmented)
- .background(Color.clear)
- .frame(width: UIScreen.main.bounds.width / 1.5, height: 40, alignment: .center)
- }
- .padding(.vertical, 1)
- }
- // @ViewBuilder private func profiles(_: GeometryProxy) -> some View {
- // let colour: Color = colorScheme == .dark ? .black : .white
- // let colourRectangle: Color = colorScheme == .dark ? .gray.opacity(0.1) : .white
- //
- // ZStack {
- // Rectangle()
- //// .fill(Color.gray.opacity(0.3))
- // .fill(colourRectangle)
- // .frame(maxHeight: 40)
- // .cornerRadius(15)
- // .padding([.leading, .trailing], 10)
- // let cancel = fetchedPercent.first?.enabled ?? false
- // HStack(spacing: cancel ? 25 : 15) {
- // Text(selectedProfile().name).foregroundColor(.secondary)
- // if cancel, selectedProfile().isOn {
- // Button { showCancelAlert.toggle() }
- // label: {
- // Image(systemName: "xmark")
- // .foregroundStyle(.secondary)
- // }
- // }
- // Button { state.showModal(for: .overrideProfilesConfig) }
- // label: {
- // Image(systemName: "person.3.sequence.fill")
- // .symbolRenderingMode(.palette)
- // .foregroundStyle(
- // !(fetchedPercent.first?.enabled ?? false) ? .green : .cyan,
- // !(fetchedPercent.first?.enabled ?? false) ? .cyan : .green,
- // .purple
- // )
- // }
- // }
- // }
- // .alert(
- // "Return to Normal?", isPresented: $showCancelAlert,
- // actions: {
- // Button("No", role: .cancel) {}
- // Button("Yes", role: .destructive) {
- // state.cancelProfile()
- // }
- // }, message: { Text("This will change settings back to your normal profile.") }
- // )
- // Rectangle().fill(colour).frame(maxHeight: 1)
- // }
- private func selectedProfile() -> (name: String, isOn: Bool) {
- var profileString = ""
- var display: Bool = false
- let duration = (fetchedPercent.first?.duration ?? 0) as Decimal
- let indefinite = fetchedPercent.first?.indefinite ?? false
- let addedMinutes = Int(duration)
- let date = fetchedPercent.first?.date ?? Date()
- if date.addingTimeInterval(addedMinutes.minutes.timeInterval) > Date() || indefinite {
- display.toggle()
- }
- if fetchedPercent.first?.enabled ?? false, !(fetchedPercent.first?.isPreset ?? false), display {
- profileString = NSLocalizedString("Custom Profile", comment: "Custom but unsaved Profile")
- } else if !(fetchedPercent.first?.enabled ?? false) || !display {
- profileString = NSLocalizedString("Normal Profile", comment: "Your normal Profile. Use a short string")
- } else {
- let id_ = fetchedPercent.first?.id ?? ""
- let profile = fetchedProfiles.filter({ $0.id == id_ }).first
- if profile != nil {
- profileString = profile?.name?.description ?? ""
- }
- }
- return (name: profileString, isOn: display)
- }
- @ViewBuilder private func bottomPanel(_: GeometryProxy) -> some View {
- let colourRectangle: Color = colorScheme == .dark ? .gray.opacity(0.2) : .white
- ZStack {
- Rectangle()
- .fill(colourRectangle)
- .frame(height: UIScreen.main.bounds.height / 13)
- .cornerRadius(15)
- .shadow(radius: 3)
- .padding([.leading, .trailing], 10)
- HStack {
- Button { state.showModal(for: .addCarbs(editMode: false, override: false)) }
- label: {
- ZStack(alignment: Alignment(horizontal: .trailing, vertical: .bottom)) {
- Image("carbs")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .foregroundColor(.loopYellow)
- .padding(8)
- if let carbsReq = state.carbsRequired {
- Text(numberFormatter.string(from: carbsReq as NSNumber)!)
- .font(.caption)
- .foregroundColor(.white)
- .padding(4)
- .background(Capsule().fill(Color.red))
- }
- }
- }.buttonStyle(.borderless)
- Spacer()
- Button {
- state.showModal(for: .bolus(
- waitForSuggestion: true,
- fetch: false
- ))
- }
- label: {
- Image("bolus")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.insulin)
- .buttonStyle(.borderless)
- Spacer()
- if state.allowManualTemp {
- Button { state.showModal(for: .manualTempBasal) }
- label: {
- Image("bolus1")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.insulin)
- .buttonStyle(.borderless)
- Spacer()
- }
- Button { state.showModal(for: .addTempTarget) }
- label: {
- Image("target")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.loopGreen)
- .buttonStyle(.borderless)
- Spacer()
- // MARK: CANCEL OF PROFILE HAS TO BE IMPLEMENTED
- // MAYBE WITH A SMALL INDICATOR AT THE SYMBOL
- Button {
- state.showModal(for: .overrideProfilesConfig)
- } label: {
- Image(systemName: "person.fill")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.purple)
- .buttonStyle(.borderless)
- Spacer()
- Button { state.showModal(for: .statistics)
- }
- label: {
- Image(systemName: "chart.xyaxis.line")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.purple)
- .buttonStyle(.borderless)
- Spacer()
- Button { state.showModal(for: .settings) }
- label: {
- Image("settings1")
- .renderingMode(.template)
- .resizable()
- .frame(width: 24, height: 24)
- .padding(8)
- }
- .foregroundColor(.loopGray)
- .buttonStyle(.borderless)
- }
- .padding(.horizontal, 24)
- }
- }
- var body: some View {
- let colourBackground: Color = colorScheme == .dark ? .black.opacity(0.5) : .gray.opacity(0.1)
- let colourChart: Color = colorScheme == .dark ? .gray.opacity(0.2) : .white
- GeometryReader { geo in
- VStack(spacing: 0) {
- ZStack {
- glucoseView
- .padding(.top, 75)
- loopView
- /// circles width is 110, loops width is 35 -> (110/2) - (35/2) = 55 - 17.5 = 37.5
- .offset(x: UIScreen.main.bounds.width * 0.43, y: -37.5)
- .padding(.top, 75)
- }
- header(geo)
- .padding(.top, 40)
- .padding([.leading, .trailing], 10)
- infoPanel
- .padding([.leading, .trailing], 10)
- .padding(.top, 25)
- RoundedRectangle(cornerRadius: 15)
- .fill(colourChart)
- .overlay(mainChart)
- .clipShape(RoundedRectangle(cornerRadius: 15))
- .shadow(radius: 3)
- .padding([.leading, .trailing], 10)
- .padding(.top, 5)
- .frame(maxHeight: UIScreen.main.bounds.height / 2.2)
- pickerPanel(geo)
- .padding(.top, 13)
- legendPanel
- .padding(.top, 10)
- bottomPanel(geo)
- .padding(.top, 3)
- }
- .edgesIgnoringSafeArea(.vertical)
- }
- .onAppear(perform: configureView)
- .navigationTitle("Home")
- .background(colourBackground)
- .navigationBarHidden(true)
- .ignoresSafeArea(.keyboard)
- .popup(isPresented: isStatusPopupPresented, alignment: .top, direction: .top) {
- popup
- .padding()
- .background(
- RoundedRectangle(cornerRadius: 8, style: .continuous)
- .fill(Color(UIColor.darkGray))
- )
- .onTapGesture {
- isStatusPopupPresented = false
- }
- .gesture(
- DragGesture(minimumDistance: 10, coordinateSpace: .local)
- .onEnded { value in
- if value.translation.height < 0 {
- isStatusPopupPresented = false
- }
- }
- )
- }
- }
- private var popup: some View {
- VStack(alignment: .leading, spacing: 4) {
- Text(state.statusTitle).font(.headline).foregroundColor(.white)
- .padding(.bottom, 4)
- if let suggestion = state.suggestion {
- TagCloudView(tags: suggestion.reasonParts).animation(.none, value: false)
- Text(suggestion.reasonConclusion.capitalizingFirstLetter()).font(.caption).foregroundColor(.white)
- } else {
- Text("No sugestion found").font(.body).foregroundColor(.white)
- }
- if let errorMessage = state.errorMessage, let date = state.errorDate {
- Text(NSLocalizedString("Error at", comment: "") + " " + dateFormatter.string(from: date))
- .foregroundColor(.white)
- .font(.headline)
- .padding(.bottom, 4)
- .padding(.top, 8)
- Text(errorMessage).font(.caption).foregroundColor(.loopRed)
- } else if let suggestion = state.suggestion, (suggestion.bg ?? 100) == 400 {
- Text("Invalid CGM reading (HIGH).").font(.callout).bold().foregroundColor(.loopRed).padding(.top, 8)
- Text("SMBs and High Temps Disabled.").font(.caption).foregroundColor(.white).padding(.bottom, 4)
- }
- }
- }
- }
- }
|