CurrentGlucoseView.swift 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. import CoreData
  2. import SwiftUI
  3. struct CurrentGlucoseView: View {
  4. let timerDate: Date
  5. let units: GlucoseUnits
  6. let alarm: GlucoseAlarm?
  7. let lowGlucose: Decimal
  8. let highGlucose: Decimal
  9. let cgmAvailable: Bool
  10. var currentGlucoseTarget: Decimal
  11. let glucoseColorScheme: GlucoseColorScheme
  12. let glucose: [GlucoseStored] // This contains the last two glucose values, no matter if its manual or a cgm reading
  13. @State private var rotationDegrees: Double = 0.0
  14. @State private var angularGradient = AngularGradient(colors: [
  15. Color(red: 0.7215686275, green: 0.3411764706, blue: 1),
  16. Color(red: 0.6235294118, green: 0.4235294118, blue: 0.9803921569),
  17. Color(red: 0.4862745098, green: 0.5450980392, blue: 0.9529411765),
  18. Color(red: 0.3411764706, green: 0.6666666667, blue: 0.9254901961),
  19. Color(red: 0.262745098, green: 0.7333333333, blue: 0.9137254902),
  20. Color(red: 0.7215686275, green: 0.3411764706, blue: 1)
  21. ], center: .center, startAngle: .degrees(270), endAngle: .degrees(-90))
  22. @Environment(\.colorScheme) var colorScheme
  23. private var deltaFormatter: NumberFormatter {
  24. let formatter = NumberFormatter()
  25. formatter.numberStyle = .decimal
  26. if units == .mmolL {
  27. formatter.maximumFractionDigits = 1
  28. formatter.minimumFractionDigits = 1
  29. formatter.roundingMode = .halfUp
  30. } else {
  31. formatter.maximumFractionDigits = 0
  32. }
  33. formatter.positivePrefix = " +"
  34. formatter.negativePrefix = " -"
  35. return formatter
  36. }
  37. var body: some View {
  38. let triangleColor = Color(red: 0.262745098, green: 0.7333333333, blue: 0.9137254902)
  39. if cgmAvailable {
  40. ZStack {
  41. TrendShape(gradient: angularGradient, color: triangleColor)
  42. .rotationEffect(.degrees(rotationDegrees))
  43. VStack(alignment: .center) {
  44. HStack {
  45. if let glucoseValue = glucose.last?.glucose {
  46. let displayGlucose = units == .mgdL ? Decimal(glucoseValue).description : Decimal(glucoseValue)
  47. .formattedAsMmolL
  48. var glucoseDisplayColor = Color.primary
  49. // TODO: workaround for now: set low value to 55, to have dynamic color shades between 55 and user-set low (approx. 70); same for high glucose
  50. let hardCodedLow = Decimal(55)
  51. let hardCodedHigh = Decimal(220)
  52. let isDynamicColorScheme = glucoseColorScheme == .dynamicColor
  53. if Decimal(glucoseValue) <= lowGlucose || Decimal(glucoseValue) >= highGlucose {
  54. glucoseDisplayColor = Trio.getDynamicGlucoseColor(
  55. glucoseValue: Decimal(glucoseValue),
  56. highGlucoseColorValue: isDynamicColorScheme ? hardCodedHigh : highGlucose,
  57. lowGlucoseColorValue: isDynamicColorScheme ? hardCodedLow : lowGlucose,
  58. targetGlucose: currentGlucoseTarget,
  59. glucoseColorScheme: glucoseColorScheme
  60. )
  61. }
  62. return Text(
  63. glucoseValue == 400 ? "HIGH" : displayGlucose
  64. )
  65. .font(.system(size: 40, weight: .bold, design: .rounded))
  66. .foregroundStyle(glucoseDisplayColor)
  67. } else {
  68. return Text("--")
  69. .font(.system(size: 40, weight: .bold, design: .rounded))
  70. .foregroundStyle(.secondary)
  71. }
  72. }
  73. HStack {
  74. let minutesAgo = -1 * (glucose.last?.date?.timeIntervalSinceNow ?? 0) / 60
  75. var minutesAgoString: String {
  76. if minutesAgo > 1 {
  77. let minuteString = Formatter.timaAgoFormatter.string(for: Double(minutesAgo)) ?? ""
  78. return minuteString + "\u{00A0}" + String(localized: "m", comment: "Abbreviation for Minutes")
  79. } else {
  80. return "<" + "\u{00A0}" + "1" + "\u{00A0}" +
  81. String(localized: "m", comment: "Abbreviation for Minutes")
  82. }
  83. }
  84. Group {
  85. Text(minutesAgoString)
  86. Text(delta)
  87. }
  88. .font(.callout).fontWeight(.bold)
  89. .foregroundStyle(colorScheme == .dark ? Color.white.opacity(0.9) : Color.secondary)
  90. }
  91. .frame(alignment: .top)
  92. }
  93. }
  94. .onChange(of: glucose.last?.directionEnum) {
  95. withAnimation {
  96. switch glucose.last?.directionEnum {
  97. case .doubleUp,
  98. .singleUp,
  99. .tripleUp:
  100. rotationDegrees = -90
  101. case .fortyFiveUp:
  102. rotationDegrees = -45
  103. case .flat:
  104. rotationDegrees = 0
  105. case .fortyFiveDown:
  106. rotationDegrees = 45
  107. case .doubleDown,
  108. .singleDown,
  109. .tripleDown:
  110. rotationDegrees = 90
  111. case nil,
  112. .notComputable,
  113. .rateOutOfRange:
  114. rotationDegrees = 0
  115. default:
  116. rotationDegrees = 0
  117. }
  118. }
  119. }
  120. } else {
  121. VStack(alignment: .center, spacing: 12) {
  122. HStack
  123. {
  124. // no cgm defined so display a generic CGM
  125. Image(systemName: "sensor.tag.radiowaves.forward.fill").font(.body).imageScale(.large)
  126. }
  127. HStack {
  128. Text("Add CGM").font(.caption).bold()
  129. }
  130. }.frame(alignment: .top)
  131. }
  132. }
  133. private var delta: String {
  134. guard glucose.count >= 2 else {
  135. return "--"
  136. }
  137. let lastGlucose = glucose.last?.glucose ?? 0
  138. let secondLastGlucose = glucose.first?.glucose ?? 0
  139. let delta = lastGlucose - secondLastGlucose
  140. let deltaAsDecimal = units == .mmolL ? Decimal(delta).asMmolL : Decimal(delta)
  141. return deltaFormatter.string(from: deltaAsDecimal as NSNumber) ?? "--"
  142. }
  143. }
  144. struct Triangle: Shape {
  145. func path(in rect: CGRect) -> Path {
  146. var path = Path()
  147. path.move(to: CGPoint(x: rect.midX, y: rect.minY + 15))
  148. path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY))
  149. path.addQuadCurve(to: CGPoint(x: rect.minX, y: rect.maxY), control: CGPoint(x: rect.midX, y: rect.midY + 10))
  150. path.closeSubpath()
  151. return path
  152. }
  153. }
  154. struct TrendShape: View {
  155. @Environment(\.colorScheme) var colorScheme
  156. let gradient: AngularGradient
  157. let color: Color
  158. var body: some View {
  159. HStack(alignment: .center) {
  160. ZStack {
  161. Group {
  162. CircleShape(gradient: gradient)
  163. TriangleShape(color: color)
  164. }.shadow(color: Color.black.opacity(colorScheme == .dark ? 0.75 : 0.33), radius: colorScheme == .dark ? 5 : 3)
  165. CircleShape(gradient: gradient)
  166. }
  167. }
  168. }
  169. }
  170. struct CircleShape: View {
  171. @Environment(\.colorScheme) var colorScheme
  172. let gradient: AngularGradient
  173. var body: some View {
  174. Circle()
  175. .stroke(gradient, lineWidth: 6)
  176. .background(Circle().fill(Color.chart))
  177. .frame(width: 130, height: 130)
  178. }
  179. }
  180. struct TriangleShape: View {
  181. let color: Color
  182. var body: some View {
  183. Triangle()
  184. .fill(color)
  185. .frame(width: 35, height: 35)
  186. .rotationEffect(.degrees(90))
  187. .offset(x: 85)
  188. }
  189. }