CurrentGlucoseView.swift 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  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 glucoseFormatter: 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. return formatter
  34. }
  35. private var deltaFormatter: NumberFormatter {
  36. let formatter = NumberFormatter()
  37. formatter.numberStyle = .decimal
  38. if units == .mmolL {
  39. formatter.maximumFractionDigits = 1
  40. formatter.minimumFractionDigits = 1
  41. formatter.roundingMode = .halfUp
  42. } else {
  43. formatter.maximumFractionDigits = 0
  44. }
  45. formatter.positivePrefix = " +"
  46. formatter.negativePrefix = " -"
  47. return formatter
  48. }
  49. private var timaAgoFormatter: NumberFormatter {
  50. let formatter = NumberFormatter()
  51. formatter.numberStyle = .decimal
  52. formatter.maximumFractionDigits = 0
  53. formatter.negativePrefix = ""
  54. return formatter
  55. }
  56. private var dateFormatter: DateFormatter {
  57. let formatter = DateFormatter()
  58. formatter.timeStyle = .short
  59. return formatter
  60. }
  61. var body: some View {
  62. let triangleColor = Color(red: 0.262745098, green: 0.7333333333, blue: 0.9137254902)
  63. if cgmAvailable {
  64. ZStack {
  65. TrendShape(gradient: angularGradient, color: triangleColor)
  66. .rotationEffect(.degrees(rotationDegrees))
  67. VStack(alignment: .center) {
  68. HStack {
  69. if let glucoseValue = glucose.last?.glucose {
  70. let displayGlucose = units == .mgdL ? Decimal(glucoseValue).description : Decimal(glucoseValue)
  71. .formattedAsMmolL
  72. // low glucose, high glucose and target is parsed in state to mmol/L; parse it back to mg/dl here for comparison
  73. let lowGlucose = units == .mgdL ? lowGlucose : lowGlucose.asMgdL
  74. let highGlucose = units == .mgdL ? highGlucose : highGlucose.asMgdL
  75. let targetGlucose = units == .mgdL ? currentGlucoseTarget : currentGlucoseTarget.asMgdL
  76. var glucoseDisplayColor = Color.primary
  77. if Decimal(glucoseValue) <= lowGlucose || Decimal(glucoseValue) >= highGlucose {
  78. glucoseDisplayColor = FreeAPS.getDynamicGlucoseColor(
  79. glucoseValue: Decimal(glucoseValue),
  80. highGlucoseColorValue: highGlucose,
  81. lowGlucoseColorValue: lowGlucose,
  82. targetGlucose: targetGlucose,
  83. glucoseColorScheme: glucoseColorScheme,
  84. offset: 20
  85. )
  86. }
  87. return Text(
  88. glucoseValue == 400 ? "HIGH" : displayGlucose
  89. )
  90. .font(.system(size: 40, weight: .bold, design: .rounded))
  91. .foregroundStyle(glucoseDisplayColor)
  92. } else {
  93. return Text("--")
  94. .font(.system(size: 40, weight: .bold, design: .rounded))
  95. .foregroundStyle(.secondary)
  96. }
  97. }
  98. HStack {
  99. let minutesAgo = -1 * (glucose.last?.date?.timeIntervalSinceNow ?? 0) / 60
  100. let text = timaAgoFormatter.string(for: Double(minutesAgo)) ?? ""
  101. Text(
  102. minutesAgo <= 1 ? "< 1 " + NSLocalizedString("min", comment: "Short form for minutes") : (
  103. text + " " +
  104. NSLocalizedString("min", comment: "Short form for minutes") + " "
  105. )
  106. )
  107. .font(.caption2).foregroundStyle(colorScheme == .dark ? Color.white.opacity(0.9) : Color.secondary)
  108. Text(
  109. delta
  110. )
  111. .font(.caption2).foregroundStyle(colorScheme == .dark ? Color.white.opacity(0.9) : Color.secondary)
  112. }.frame(alignment: .top)
  113. }
  114. }
  115. .onChange(of: glucose.last?.directionEnum) {
  116. withAnimation {
  117. switch glucose.last?.directionEnum {
  118. case .doubleUp,
  119. .singleUp,
  120. .tripleUp:
  121. rotationDegrees = -90
  122. case .fortyFiveUp:
  123. rotationDegrees = -45
  124. case .flat:
  125. rotationDegrees = 0
  126. case .fortyFiveDown:
  127. rotationDegrees = 45
  128. case .doubleDown,
  129. .singleDown,
  130. .tripleDown:
  131. rotationDegrees = 90
  132. case nil,
  133. .notComputable,
  134. .rateOutOfRange:
  135. rotationDegrees = 0
  136. default:
  137. rotationDegrees = 0
  138. }
  139. }
  140. }
  141. } else {
  142. VStack(alignment: .center, spacing: 12) {
  143. HStack
  144. {
  145. // no cgm defined so display a generic CGM
  146. Image(systemName: "sensor.tag.radiowaves.forward.fill").font(.body).imageScale(.large)
  147. }
  148. HStack {
  149. Text("Add CGM").font(.caption).bold()
  150. }
  151. }.frame(alignment: .top)
  152. }
  153. }
  154. private var delta: String {
  155. guard glucose.count >= 2 else {
  156. return "--"
  157. }
  158. let lastGlucose = glucose.last?.glucose ?? 0
  159. let secondLastGlucose = glucose.first?.glucose ?? 0
  160. let delta = lastGlucose - secondLastGlucose
  161. let deltaAsDecimal = units == .mmolL ? Decimal(delta).asMmolL : Decimal(delta)
  162. return deltaFormatter.string(from: deltaAsDecimal as NSNumber) ?? "--"
  163. }
  164. // var glucoseDisplayColor: Color {
  165. // guard let lastGlucose = glucose.last?.glucose else { return .primary }
  166. //
  167. // // low and high glucose is parsed in state to mmol/L; parse it back to mg/dl here for comparison
  168. // let lowGlucose = units == .mgdL ? lowGlucose : lowGlucose.asMgdL
  169. // let highGlucose = units == .mgdL ? highGlucose : highGlucose.asMgdL
  170. //
  171. // // Ensure the thresholds are logical
  172. // guard lowGlucose < highGlucose else { return .primary }
  173. //
  174. // guard Decimal(lastGlucose) <= lowGlucose && Decimal(lastGlucose) >= highGlucose else { return .primary }
  175. //
  176. // return FreeAPS.getDynamicGlucoseColor(
  177. // glucoseValue: Decimal(lastGlucose),
  178. // highGlucoseColorValue: highGlucose,
  179. // lowGlucoseColorValue: lowGlucose,
  180. // targetGlucose: currentGlucoseTarget,
  181. // glucoseColorScheme: glucoseColorScheme,
  182. // offset: 20
  183. // )
  184. // }
  185. }
  186. struct Triangle: Shape {
  187. func path(in rect: CGRect) -> Path {
  188. var path = Path()
  189. path.move(to: CGPoint(x: rect.midX, y: rect.minY + 15))
  190. path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY))
  191. path.addQuadCurve(to: CGPoint(x: rect.minX, y: rect.maxY), control: CGPoint(x: rect.midX, y: rect.midY + 10))
  192. path.closeSubpath()
  193. return path
  194. }
  195. }
  196. struct TrendShape: View {
  197. @Environment(\.colorScheme) var colorScheme
  198. let gradient: AngularGradient
  199. let color: Color
  200. var body: some View {
  201. HStack(alignment: .center) {
  202. ZStack {
  203. Group {
  204. CircleShape(gradient: gradient)
  205. TriangleShape(color: color)
  206. }.shadow(color: Color.black.opacity(colorScheme == .dark ? 0.75 : 0.33), radius: colorScheme == .dark ? 5 : 3)
  207. CircleShape(gradient: gradient)
  208. }
  209. }
  210. }
  211. }
  212. struct CircleShape: View {
  213. @Environment(\.colorScheme) var colorScheme
  214. let gradient: AngularGradient
  215. var body: some View {
  216. Circle()
  217. .stroke(gradient, lineWidth: 6)
  218. .background(Circle().fill(Color.chart))
  219. .frame(width: 130, height: 130)
  220. }
  221. }
  222. struct TriangleShape: View {
  223. let color: Color
  224. var body: some View {
  225. Triangle()
  226. .fill(color)
  227. .frame(width: 35, height: 35)
  228. .rotationEffect(.degrees(90))
  229. .offset(x: 85)
  230. }
  231. }