CurrentGlucoseView.swift 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. import CoreData
  2. import SwiftUI
  3. struct CurrentGlucoseView: View {
  4. @Binding var timerDate: Date
  5. @Binding var units: GlucoseUnits
  6. @Binding var alarm: GlucoseAlarm?
  7. @Binding var lowGlucose: Decimal
  8. @Binding var highGlucose: Decimal
  9. @Binding var cgmAvailable: Bool
  10. var glucose: [GlucoseStored] // This contains the last two glucose values, no matter if its manual or a cgm reading
  11. @State private var rotationDegrees: Double = 0.0
  12. @State private var angularGradient = AngularGradient(colors: [
  13. Color(red: 0.7215686275, green: 0.3411764706, blue: 1),
  14. Color(red: 0.6235294118, green: 0.4235294118, blue: 0.9803921569),
  15. Color(red: 0.4862745098, green: 0.5450980392, blue: 0.9529411765),
  16. Color(red: 0.3411764706, green: 0.6666666667, blue: 0.9254901961),
  17. Color(red: 0.262745098, green: 0.7333333333, blue: 0.9137254902),
  18. Color(red: 0.7215686275, green: 0.3411764706, blue: 1)
  19. ], center: .center, startAngle: .degrees(270), endAngle: .degrees(-90))
  20. @Environment(\.colorScheme) var colorScheme
  21. private var glucoseFormatter: NumberFormatter {
  22. let formatter = NumberFormatter()
  23. formatter.numberStyle = .decimal
  24. if units == .mmolL {
  25. formatter.maximumFractionDigits = 1
  26. formatter.minimumFractionDigits = 1
  27. formatter.roundingMode = .halfUp
  28. } else {
  29. formatter.maximumFractionDigits = 0
  30. }
  31. return formatter
  32. }
  33. private var deltaFormatter: NumberFormatter {
  34. let formatter = NumberFormatter()
  35. formatter.numberStyle = .decimal
  36. if units == .mmolL {
  37. formatter.maximumFractionDigits = 1
  38. formatter.minimumFractionDigits = 1
  39. formatter.roundingMode = .halfUp
  40. } else {
  41. formatter.maximumFractionDigits = 0
  42. }
  43. formatter.positivePrefix = " +"
  44. formatter.negativePrefix = " -"
  45. return formatter
  46. }
  47. private var timaAgoFormatter: NumberFormatter {
  48. let formatter = NumberFormatter()
  49. formatter.numberStyle = .decimal
  50. formatter.maximumFractionDigits = 0
  51. formatter.negativePrefix = ""
  52. return formatter
  53. }
  54. private var dateFormatter: DateFormatter {
  55. let formatter = DateFormatter()
  56. formatter.timeStyle = .short
  57. return formatter
  58. }
  59. var body: some View {
  60. let triangleColor = Color(red: 0.262745098, green: 0.7333333333, blue: 0.9137254902)
  61. if cgmAvailable {
  62. ZStack {
  63. TrendShape(gradient: angularGradient, color: triangleColor)
  64. .rotationEffect(.degrees(rotationDegrees))
  65. VStack(alignment: .center) {
  66. HStack {
  67. if let glucoseValue = glucose.last?.glucose {
  68. let displayGlucose = units == .mgdL ? Decimal(glucoseValue).description : Decimal(glucoseValue)
  69. .formattedAsMmolL
  70. Text(
  71. glucoseValue == 400 ? "HIGH" : displayGlucose
  72. )
  73. .font(.system(size: 40, weight: .bold, design: .rounded))
  74. .foregroundColor(alarm == nil ? glucoseDisplayColor : .loopRed)
  75. } else {
  76. Text("--")
  77. .font(.system(size: 40, weight: .bold, design: .rounded))
  78. .foregroundColor(.secondary)
  79. }
  80. }
  81. HStack {
  82. let minutesAgo = -1 * (glucose.last?.date?.timeIntervalSinceNow ?? 0) / 60
  83. let text = timaAgoFormatter.string(for: Double(minutesAgo)) ?? ""
  84. Text(
  85. minutesAgo <= 1 ? "< 1 " + NSLocalizedString("min", comment: "Short form for minutes") : (
  86. text + " " +
  87. NSLocalizedString("min", comment: "Short form for minutes") + " "
  88. )
  89. )
  90. .font(.caption2).foregroundColor(colorScheme == .dark ? Color.white.opacity(0.9) : Color.secondary)
  91. Text(
  92. delta
  93. )
  94. .font(.caption2).foregroundColor(colorScheme == .dark ? Color.white.opacity(0.9) : Color.secondary)
  95. }.frame(alignment: .top)
  96. }
  97. }
  98. .onChange(of: glucose.last?.directionEnum) { newDirection in
  99. withAnimation {
  100. switch newDirection {
  101. case .doubleUp,
  102. .singleUp,
  103. .tripleUp:
  104. rotationDegrees = -90
  105. case .fortyFiveUp:
  106. rotationDegrees = -45
  107. case .flat:
  108. rotationDegrees = 0
  109. case .fortyFiveDown:
  110. rotationDegrees = 45
  111. case .doubleDown,
  112. .singleDown,
  113. .tripleDown:
  114. rotationDegrees = 90
  115. case nil,
  116. .notComputable,
  117. .rateOutOfRange:
  118. rotationDegrees = 0
  119. default:
  120. rotationDegrees = 0
  121. }
  122. }
  123. }
  124. } else {
  125. VStack(alignment: .center, spacing: 12) {
  126. HStack
  127. {
  128. // no cgm defined so display a generic CGM
  129. Image(systemName: "sensor.tag.radiowaves.forward.fill").font(.body).imageScale(.large)
  130. }
  131. HStack {
  132. Text("Add CGM").font(.caption).bold()
  133. }
  134. }.frame(alignment: .top)
  135. }
  136. }
  137. private var delta: String {
  138. guard glucose.count >= 2 else {
  139. return "--"
  140. }
  141. let lastGlucose = glucose.last?.glucose ?? 0
  142. let secondLastGlucose = glucose.first?.glucose ?? 0
  143. let delta = lastGlucose - secondLastGlucose
  144. let deltaAsDecimal = units == .mmolL ? Decimal(delta).asMmolL : Decimal(delta)
  145. return deltaFormatter.string(from: deltaAsDecimal as NSNumber) ?? "--"
  146. }
  147. var glucoseDisplayColor: Color {
  148. guard let lastGlucose = glucose.last?.glucose else { return .primary }
  149. // Convert the lastest glucose value to Int for comparison
  150. let whichGlucose = Int(lastGlucose)
  151. // Define default color based on the color scheme
  152. let defaultColor: Color = colorScheme == .dark ? .white : .black
  153. // low and high glucose is parsed in state to mmol/L; parse it back to mg/dl here for comparison
  154. let lowGlucose = units == .mgdL ? lowGlucose : lowGlucose.asMgdL
  155. let highGlucose = units == .mgdL ? highGlucose : highGlucose.asMgdL
  156. // Ensure the thresholds are logical
  157. guard lowGlucose < highGlucose else { return .primary }
  158. // Perform range checks using Int converted values
  159. switch whichGlucose {
  160. case 0 ..< Int(lowGlucose):
  161. return .loopRed
  162. case Int(lowGlucose) ..< Int(highGlucose):
  163. return defaultColor
  164. case Int(highGlucose)...:
  165. return .loopYellow
  166. default:
  167. return defaultColor
  168. }
  169. }
  170. }
  171. struct Triangle: Shape {
  172. func path(in rect: CGRect) -> Path {
  173. var path = Path()
  174. path.move(to: CGPoint(x: rect.midX, y: rect.minY + 15))
  175. path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY))
  176. path.addQuadCurve(to: CGPoint(x: rect.minX, y: rect.maxY), control: CGPoint(x: rect.midX, y: rect.midY + 10))
  177. path.closeSubpath()
  178. return path
  179. }
  180. }
  181. struct TrendShape: View {
  182. @Environment(\.colorScheme) var colorScheme
  183. let gradient: AngularGradient
  184. let color: Color
  185. var body: some View {
  186. HStack(alignment: .center) {
  187. ZStack {
  188. Group {
  189. CircleShape(gradient: gradient)
  190. TriangleShape(color: color)
  191. }.shadow(color: Color.black.opacity(colorScheme == .dark ? 0.75 : 0.33), radius: colorScheme == .dark ? 5 : 3)
  192. CircleShape(gradient: gradient)
  193. }
  194. }
  195. }
  196. }
  197. struct CircleShape: View {
  198. @Environment(\.colorScheme) var colorScheme
  199. let gradient: AngularGradient
  200. var body: some View {
  201. Circle()
  202. .stroke(gradient, lineWidth: 6)
  203. .background(Circle().fill(Color.chart))
  204. .frame(width: 130, height: 130)
  205. }
  206. }
  207. struct TriangleShape: View {
  208. let color: Color
  209. var body: some View {
  210. Triangle()
  211. .fill(color)
  212. .frame(width: 35, height: 35)
  213. .rotationEffect(.degrees(90))
  214. .offset(x: 85)
  215. }
  216. }