CurrentGlucoseView.swift 7.6 KB

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