CurrentGlucoseView.swift 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. import SwiftUI
  2. struct CurrentGlucoseView: View {
  3. @Binding var recentGlucose: BloodGlucose?
  4. @Binding var timerDate: Date
  5. @Binding var delta: Int?
  6. @Binding var units: GlucoseUnits
  7. @Binding var alarm: GlucoseAlarm?
  8. @Binding var lowGlucose: Decimal
  9. @Binding var highGlucose: Decimal
  10. @State private var rotationDegrees: Double = 0.0
  11. @State private var angularGradient: AngularGradient = defaultGradient()
  12. @Environment(\.colorScheme) var colorScheme
  13. private var glucoseFormatter: NumberFormatter {
  14. let formatter = NumberFormatter()
  15. formatter.numberStyle = .decimal
  16. formatter.maximumFractionDigits = 0
  17. if units == .mmolL {
  18. formatter.minimumFractionDigits = 1
  19. formatter.maximumFractionDigits = 1
  20. }
  21. formatter.roundingMode = .halfUp
  22. return formatter
  23. }
  24. private var deltaFormatter: NumberFormatter {
  25. let formatter = NumberFormatter()
  26. formatter.numberStyle = .decimal
  27. formatter.maximumFractionDigits = 1
  28. formatter.positivePrefix = " +"
  29. formatter.negativePrefix = " -"
  30. return formatter
  31. }
  32. private var timaAgoFormatter: NumberFormatter {
  33. let formatter = NumberFormatter()
  34. formatter.numberStyle = .decimal
  35. formatter.maximumFractionDigits = 0
  36. formatter.negativePrefix = ""
  37. return formatter
  38. }
  39. private var dateFormatter: DateFormatter {
  40. let formatter = DateFormatter()
  41. formatter.timeStyle = .short
  42. return formatter
  43. }
  44. var body: some View {
  45. // let triangleColor = Color(red: 0.729, green: 0.337, blue: 1)
  46. let triangleColor = Color(red: 0.263, green: 0.733, blue: 0.914)
  47. ZStack {
  48. TrendShape(gradient: angularGradient, color: triangleColor)
  49. .rotationEffect(.degrees(rotationDegrees))
  50. VStack(alignment: .center) {
  51. HStack {
  52. Text(
  53. (recentGlucose?.glucose ?? 100) == 400 ? "HIGH" : recentGlucose?.glucose
  54. .map {
  55. glucoseFormatter
  56. .string(from: Double(units == .mmolL ? $0.asMmolL : Decimal($0)) as NSNumber)! }
  57. ?? "--"
  58. )
  59. .font(.system(size: 40, weight: .bold))
  60. .foregroundColor(alarm == nil ? colourGlucoseText : .loopRed)
  61. }
  62. HStack {
  63. let minutesAgo = -1 * (recentGlucose?.dateString.timeIntervalSinceNow ?? 0) / 60
  64. let text = timaAgoFormatter.string(for: Double(minutesAgo)) ?? ""
  65. Text(
  66. minutesAgo <= 1 ? "< 1 " + NSLocalizedString("min", comment: "Short form for minutes") : (
  67. text + " " +
  68. NSLocalizedString("min", comment: "Short form for minutes") + " "
  69. )
  70. )
  71. .font(.caption2).foregroundColor(.secondary)
  72. Text(
  73. delta
  74. .map {
  75. deltaFormatter.string(from: Double(units == .mmolL ? $0.asMmolL : Decimal($0)) as NSNumber)!
  76. } ?? "--"
  77. )
  78. .font(.caption2).foregroundColor(.secondary)
  79. }.frame(alignment: .top)
  80. }
  81. }
  82. .onChange(of: recentGlucose?.direction) { newDirection in
  83. withAnimation {
  84. switch newDirection {
  85. case .doubleUp,
  86. .singleUp,
  87. .tripleUp:
  88. rotationDegrees = -90
  89. setupAngularGradient(startAngle: -135, endAngle: 45)
  90. case .fortyFiveUp:
  91. rotationDegrees = -45
  92. setupAngularGradient(startAngle: 0, endAngle: 90)
  93. case .flat:
  94. rotationDegrees = 0
  95. setupAngularGradient(startAngle: -45, endAngle: 135)
  96. case .fortyFiveDown:
  97. rotationDegrees = 45
  98. setupAngularGradient(startAngle: 0, endAngle: 180)
  99. case .doubleDown,
  100. .singleDown,
  101. .tripleDown:
  102. rotationDegrees = 90
  103. setupAngularGradient(startAngle: 45, endAngle: 225)
  104. case .none,
  105. .notComputable,
  106. .rateOutOfRange:
  107. rotationDegrees = 0
  108. setupAngularGradient(startAngle: -45, endAngle: 135)
  109. @unknown default:
  110. rotationDegrees = 0
  111. setupAngularGradient(startAngle: -45, endAngle: 135)
  112. }
  113. }
  114. }
  115. }
  116. private func setupAngularGradient(startAngle: Double, endAngle: Double) {
  117. angularGradient = AngularGradient(colors: [
  118. Color(red: 0.729, green: 0.337, blue: 1),
  119. Color(red: 0.263, green: 0.733, blue: 0.914),
  120. Color(red: 0.263, green: 0.733, blue: 0.914),
  121. Color(red: 0.263, green: 0.733, blue: 0.914),
  122. Color(red: 0.263, green: 0.733, blue: 0.914),
  123. Color(red: 0.729, green: 0.337, blue: 1)
  124. ], center: .center, startAngle: .degrees(startAngle), endAngle: .degrees(endAngle))
  125. }
  126. private static func defaultGradient(startAngle: Double = -45, endAngle: Double = 135) -> AngularGradient {
  127. AngularGradient(colors: [
  128. Color(red: 0.729, green: 0.337, blue: 1),
  129. Color(red: 0.263, green: 0.733, blue: 0.914),
  130. Color(red: 0.263, green: 0.733, blue: 0.914),
  131. Color(red: 0.263, green: 0.733, blue: 0.914),
  132. Color(red: 0.263, green: 0.733, blue: 0.914),
  133. Color(red: 0.729, green: 0.337, blue: 1)
  134. ], center: .center, startAngle: .degrees(startAngle), endAngle: .degrees(endAngle))
  135. }
  136. var colourGlucoseText: Color {
  137. let whichGlucose = recentGlucose?.glucose ?? 0
  138. let defaultColor: Color = colorScheme == .dark ? .white : .black
  139. guard lowGlucose < highGlucose else { return .primary }
  140. switch whichGlucose {
  141. case 0 ..< Int(lowGlucose):
  142. return .loopRed
  143. case Int(lowGlucose) ..< Int(highGlucose):
  144. return defaultColor
  145. case Int(highGlucose)...:
  146. return .loopYellow
  147. default:
  148. return defaultColor
  149. }
  150. }
  151. }
  152. struct Triangle: Shape {
  153. func path(in rect: CGRect) -> Path {
  154. var path = Path()
  155. let cornerRadius: CGFloat = 8
  156. path.move(to: CGPoint(x: rect.midX, y: rect.minY))
  157. path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY - cornerRadius))
  158. path.addQuadCurve(to: CGPoint(x: rect.minX, y: rect.maxY - cornerRadius), control: CGPoint(x: rect.midX, y: rect.maxY))
  159. path.closeSubpath()
  160. return path
  161. }
  162. }
  163. struct TrendShape: View {
  164. let gradient: AngularGradient
  165. let color: Color
  166. var body: some View {
  167. HStack(alignment: .center) {
  168. ZStack {
  169. CircleShape(gradient: gradient)
  170. TriangleShape(color: color)
  171. }
  172. }
  173. }
  174. }
  175. struct CircleShape: View {
  176. @Environment(\.colorScheme) var colorScheme
  177. let gradient: AngularGradient
  178. var body: some View {
  179. let colorBackground: Color = colorScheme == .dark ? .black.opacity(0.8) : .white
  180. Circle()
  181. .stroke(gradient, lineWidth: 10)
  182. .shadow(
  183. color: colorScheme == .dark ? Color(red: 0.02745098039, green: 0.1098039216, blue: 0.1411764706) :
  184. Color.black.opacity(0.33),
  185. radius: 3
  186. )
  187. .background(Circle().fill(colorBackground))
  188. .frame(width: 110, height: 110)
  189. }
  190. }
  191. struct TriangleShape: View {
  192. let color: Color
  193. var body: some View {
  194. Triangle()
  195. .fill(color)
  196. .frame(width: 30, height: 30)
  197. .rotationEffect(.degrees(90))
  198. .offset(x: 65)
  199. }
  200. }