CurrentGlucoseView.swift 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  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 var gradientColor = LinearGradient(
  12. gradient: Gradient(colors: [
  13. Color(red: 0.263, green: 0.733, blue: 0.914),
  14. Color(red: 0.729, green: 0.337, blue: 1)
  15. ]),
  16. startPoint: .leading,
  17. endPoint: .trailing
  18. )
  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 colourGlucoseText: Color = colorScheme == .dark ? .white : .black
  53. let triangleColor = Color(red: 0.729, green: 0.337, blue: 1)
  54. ZStack {
  55. TrendShape(gradient: gradientColor, color: triangleColor)
  56. .rotationEffect(.degrees(rotationDegrees))
  57. VStack(alignment: .center) {
  58. HStack {
  59. Text(
  60. (recentGlucose?.glucose ?? 100) == 400 ? "HIGH" : recentGlucose?.glucose
  61. .map {
  62. glucoseFormatter
  63. .string(from: Double(units == .mmolL ? $0.asMmolL : Decimal($0)) as NSNumber)! }
  64. ?? "--"
  65. )
  66. .font(.system(size: 40, weight: .bold))
  67. .foregroundColor(alarm == nil ? colourGlucoseText : .loopRed)
  68. }
  69. HStack {
  70. let minutesAgo = -1 * (recentGlucose?.dateString.timeIntervalSinceNow ?? 0) / 60
  71. let text = timaAgoFormatter.string(for: Double(minutesAgo)) ?? ""
  72. Text(
  73. minutesAgo <= 1 ? "< 1 " + NSLocalizedString("min", comment: "Short form for minutes") : (
  74. text + " " +
  75. NSLocalizedString("min", comment: "Short form for minutes") + " "
  76. )
  77. )
  78. .font(.caption2).foregroundColor(.secondary)
  79. Text(
  80. delta
  81. .map {
  82. deltaFormatter.string(from: Double(units == .mmolL ? $0.asMmolL : Decimal($0)) as NSNumber)!
  83. } ?? "--"
  84. )
  85. .font(.caption2).foregroundColor(.secondary)
  86. }.frame(alignment: .top)
  87. }
  88. }
  89. .onChange(of: recentGlucose?.direction) { newDirection in
  90. withAnimation {
  91. switch newDirection {
  92. case .doubleUp,
  93. .singleUp,
  94. .tripleUp:
  95. rotationDegrees = -90
  96. gradientColor = LinearGradient(
  97. gradient: Gradient(colors: [
  98. Color(red: 0.263, green: 0.733, blue: 0.914),
  99. Color(red: 0.729, green: 0.337, blue: 1)
  100. ]),
  101. startPoint: .bottom,
  102. endPoint: .top
  103. )
  104. case .fortyFiveUp:
  105. rotationDegrees = -45
  106. gradientColor = LinearGradient(
  107. gradient: Gradient(colors: [
  108. Color(red: 0.263, green: 0.733, blue: 0.914),
  109. Color(red: 0.729, green: 0.337, blue: 1)
  110. ]),
  111. startPoint: .bottomLeading,
  112. endPoint: .topTrailing
  113. )
  114. case .flat:
  115. rotationDegrees = 0
  116. gradientColor = LinearGradient(
  117. gradient: Gradient(colors: [
  118. Color(red: 0.263, green: 0.733, blue: 0.914),
  119. Color(red: 0.729, green: 0.337, blue: 1)
  120. ]),
  121. startPoint: .leading,
  122. endPoint: .trailing
  123. )
  124. case .fortyFiveDown:
  125. rotationDegrees = 45
  126. gradientColor = LinearGradient(
  127. gradient: Gradient(colors: [
  128. Color(red: 0.263, green: 0.733, blue: 0.914),
  129. Color(red: 0.729, green: 0.337, blue: 1)
  130. ]),
  131. startPoint: .topLeading,
  132. endPoint: .bottomTrailing
  133. )
  134. case .doubleDown,
  135. .singleDown,
  136. .tripleDown:
  137. rotationDegrees = 90
  138. gradientColor = LinearGradient(
  139. gradient: Gradient(colors: [
  140. Color(red: 0.263, green: 0.733, blue: 0.914),
  141. Color(red: 0.729, green: 0.337, blue: 1)
  142. ]),
  143. startPoint: .top,
  144. endPoint: .bottom
  145. )
  146. case .none,
  147. .notComputable,
  148. .rateOutOfRange:
  149. rotationDegrees = 0
  150. gradientColor = LinearGradient(
  151. gradient: Gradient(colors: [
  152. Color(red: 0.263, green: 0.733, blue: 0.914),
  153. Color(red: 0.729, green: 0.337, blue: 1)
  154. ]),
  155. startPoint: .leading,
  156. endPoint: .trailing
  157. )
  158. @unknown default:
  159. rotationDegrees = 0
  160. gradientColor = LinearGradient(
  161. gradient: Gradient(colors: [
  162. Color(red: 0.263, green: 0.733, blue: 0.914),
  163. Color(red: 0.729, green: 0.337, blue: 1)
  164. ]),
  165. startPoint: .leading,
  166. endPoint: .trailing
  167. )
  168. }
  169. }
  170. }
  171. }
  172. var colorOfGlucose: Color {
  173. let whichGlucose = recentGlucose?.glucose ?? 0
  174. guard lowGlucose < highGlucose else { return .primary }
  175. switch whichGlucose {
  176. case 0 ..< Int(lowGlucose):
  177. return .loopRed
  178. case Int(lowGlucose) ..< Int(highGlucose):
  179. return .loopGreen
  180. case Int(highGlucose)...:
  181. return .loopYellow
  182. default:
  183. return .loopYellow
  184. }
  185. }
  186. }
  187. struct Triangle: Shape {
  188. func path(in rect: CGRect) -> Path {
  189. var path = Path()
  190. let cornerRadius: CGFloat = 8
  191. path.move(to: CGPoint(x: rect.midX, y: rect.minY))
  192. path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY - cornerRadius))
  193. path.addQuadCurve(to: CGPoint(x: rect.minX, y: rect.maxY - cornerRadius), control: CGPoint(x: rect.midX, y: rect.maxY))
  194. path.closeSubpath()
  195. return path
  196. }
  197. }
  198. struct TrendShape: View {
  199. let gradient: LinearGradient
  200. let color: Color
  201. var body: some View {
  202. HStack(alignment: .center) {
  203. ZStack {
  204. CircleShape(gradient: gradient)
  205. TriangleShape(color: color)
  206. }
  207. }
  208. }
  209. }
  210. struct CircleShape: View {
  211. @Environment(\.colorScheme) var colorScheme
  212. let gradient: LinearGradient
  213. var body: some View {
  214. let colorBackground: Color = colorScheme == .dark ? .black.opacity(0.8) : .white
  215. Circle()
  216. .stroke(gradient, lineWidth: 10)
  217. .shadow(radius: 3)
  218. .background(Circle().fill(colorBackground))
  219. .frame(width: 110, height: 110)
  220. }
  221. }
  222. struct TriangleShape: View {
  223. let color: Color
  224. var body: some View {
  225. Triangle()
  226. .fill(color)
  227. .frame(width: 30, height: 30)
  228. .rotationEffect(.degrees(90))
  229. .offset(x: 65)
  230. }
  231. }