CurrentGlucoseView.swift 8.9 KB

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