PumpView.swift 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. import SwiftUI
  2. struct PumpView: View {
  3. @Binding var reservoir: Decimal?
  4. @Binding var battery: Battery?
  5. @Binding var name: String
  6. @Binding var expiresAtDate: Date?
  7. @Binding var timerDate: Date
  8. private var reservoirFormatter: NumberFormatter {
  9. let formatter = NumberFormatter()
  10. formatter.numberStyle = .decimal
  11. formatter.maximumFractionDigits = 0
  12. return formatter
  13. }
  14. private var batteryFormatter: NumberFormatter {
  15. let formatter = NumberFormatter()
  16. formatter.numberStyle = .percent
  17. return formatter
  18. }
  19. var body: some View {
  20. VStack(alignment: .leading, spacing: 12) {
  21. if let reservoir = reservoir {
  22. HStack {
  23. Image(systemName: "drop.fill")
  24. .resizable()
  25. .aspectRatio(contentMode: .fit)
  26. .frame(height: 8)
  27. .foregroundColor(reservoirColor)
  28. if reservoir == 0xDEAD_BEEF {
  29. Text("50+ " + NSLocalizedString("U", comment: "Insulin unit")).font(.system(size: 12, weight: .bold))
  30. } else {
  31. Text(
  32. reservoirFormatter
  33. .string(from: reservoir as NSNumber)! + NSLocalizedString(" U", comment: "Insulin unit")
  34. )
  35. .font(.system(size: 12, weight: .bold))
  36. }
  37. }
  38. }
  39. if let battery = battery, battery.display ?? false, expiresAtDate == nil {
  40. HStack {
  41. Image(systemName: "battery.100")
  42. .resizable()
  43. .aspectRatio(contentMode: .fit)
  44. .frame(height: 8)
  45. .foregroundColor(batteryColor)
  46. Text("\(Int(battery.percent ?? 100)) %").font(.system(size: 12, weight: .bold))
  47. }
  48. }
  49. if let date = expiresAtDate {
  50. HStack {
  51. Image(systemName: "stopwatch.fill")
  52. .resizable()
  53. .aspectRatio(contentMode: .fit)
  54. .frame(height: 8)
  55. .foregroundColor(timerColor)
  56. Text(remainingTimeString(time: date.timeIntervalSince(timerDate))).font(.system(size: 12, weight: .bold))
  57. }
  58. }
  59. }
  60. }
  61. private func remainingTimeString(time: TimeInterval) -> String {
  62. guard time > 0 else {
  63. return NSLocalizedString("Replace pod", comment: "View/Header when pod expired")
  64. }
  65. var time = time
  66. let days = Int(time / 1.days.timeInterval)
  67. time -= days.days.timeInterval
  68. let hours = Int(time / 1.hours.timeInterval)
  69. time -= hours.hours.timeInterval
  70. let minutes = Int(time / 1.minutes.timeInterval)
  71. if days >= 1 {
  72. return "\(days)" + NSLocalizedString("d", comment: "abbreviation for days") + " \(hours)" +
  73. NSLocalizedString("h", comment: "abbreviation for hours")
  74. }
  75. if hours >= 1 {
  76. return "\(hours)" + NSLocalizedString("h", comment: "abbreviation for hours")
  77. }
  78. return "\(minutes)" + NSLocalizedString("m", comment: "abbreviation for minutes")
  79. }
  80. private var batteryColor: Color {
  81. guard let battery = battery, let percent = battery.percent else {
  82. return .gray
  83. }
  84. switch percent {
  85. case ...10:
  86. return .loopRed
  87. case ...20:
  88. return .loopYellow
  89. default:
  90. return .loopGreen
  91. }
  92. }
  93. private var reservoirColor: Color {
  94. guard let reservoir = reservoir else {
  95. return .gray
  96. }
  97. switch reservoir {
  98. case ...10:
  99. return .loopRed
  100. case ...30:
  101. return .loopYellow
  102. default:
  103. return .insulin
  104. }
  105. }
  106. private var timerColor: Color {
  107. guard let expisesAt = expiresAtDate else {
  108. return .gray
  109. }
  110. let time = expisesAt.timeIntervalSince(timerDate)
  111. switch time {
  112. case ...8.hours.timeInterval:
  113. return .loopRed
  114. case ...1.days.timeInterval:
  115. return .loopYellow
  116. default:
  117. return .loopGreen
  118. }
  119. }
  120. }