CurrentGlucoseView.swift 8.1 KB

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