CurrentGlucoseView.swift 7.8 KB

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