WatchStateModel.swift 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. import Combine
  2. import Foundation
  3. import SwiftUI
  4. import WatchConnectivity
  5. class WatchStateModel: NSObject, ObservableObject {
  6. var session: WCSession
  7. @Published var glucose = "00"
  8. @Published var trend = "→"
  9. @Published var delta = "+00"
  10. @Published var eventualBG = ""
  11. @Published var lastLoopDate: Date?
  12. @Published var glucoseDate: Date?
  13. @Published var bolusIncrement: Decimal?
  14. @Published var maxCOB: Decimal?
  15. @Published var maxBolus: Decimal?
  16. @Published var bolusRecommended: Decimal?
  17. @Published var carbsRequired: Decimal?
  18. @Published var iob: Decimal?
  19. @Published var cob: Decimal?
  20. @Published var tempTargets: [TempTargetWatchPreset] = []
  21. @Published var bolusAfterCarbs = true
  22. @Published var isCarbsViewActive = false
  23. @Published var isTempTargetViewActive = false
  24. @Published var isBolusViewActive = false
  25. @Published var isConfirmationViewActive = false
  26. @Published var isConfirmationBolusViewActive = false
  27. @Published var confirmationSuccess: Bool?
  28. @Published var lastUpdate: Date = .distantPast
  29. @Published var timerDate = Date()
  30. @Published var pendingBolus: Double?
  31. private var lifetime = Set<AnyCancellable>()
  32. let timer = Timer.publish(every: 10, on: .main, in: .common).autoconnect()
  33. init(session: WCSession = .default) {
  34. self.session = session
  35. super.init()
  36. session.delegate = self
  37. session.activate()
  38. }
  39. func addCarbs(_ carbs: Int) {
  40. confirmationSuccess = nil
  41. isConfirmationViewActive = true
  42. isCarbsViewActive = false
  43. session.sendMessage(["carbs": carbs], replyHandler: { reply in
  44. self.completionHandler(reply)
  45. if let ok = reply["confirmation"] as? Bool, ok, self.bolusAfterCarbs {
  46. DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
  47. self.isBolusViewActive = true
  48. }
  49. }
  50. }) { error in
  51. print(error.localizedDescription)
  52. DispatchQueue.main.async {
  53. self.confirmation(false)
  54. }
  55. }
  56. }
  57. func enactTempTarget(id: String) {
  58. confirmationSuccess = nil
  59. isConfirmationViewActive = true
  60. isTempTargetViewActive = false
  61. session.sendMessage(["tempTarget": id], replyHandler: completionHandler) { error in
  62. print(error.localizedDescription)
  63. DispatchQueue.main.async {
  64. self.confirmation(false)
  65. }
  66. }
  67. }
  68. func addBolus(amount: Double) {
  69. isBolusViewActive = false
  70. pendingBolus = amount
  71. isConfirmationBolusViewActive = true
  72. }
  73. func enactBolus() {
  74. isConfirmationBolusViewActive = false
  75. guard let amount = pendingBolus else { return }
  76. confirmationSuccess = nil
  77. isConfirmationViewActive = true
  78. session.sendMessage(["bolus": amount], replyHandler: completionHandler) { error in
  79. print(error.localizedDescription)
  80. DispatchQueue.main.async {
  81. self.confirmation(false)
  82. }
  83. }
  84. }
  85. func requestState() {
  86. guard session.activationState == .activated else {
  87. session.activate()
  88. return
  89. }
  90. session.sendMessage(["stateRequest": true], replyHandler: nil) { error in
  91. print("WatchStateModel error: " + error.localizedDescription)
  92. }
  93. }
  94. private func completionHandler(_ reply: [String: Any]) {
  95. if let ok = reply["confirmation"] as? Bool {
  96. DispatchQueue.main.async {
  97. self.confirmation(ok)
  98. }
  99. } else {
  100. DispatchQueue.main.async {
  101. self.confirmation(false)
  102. }
  103. }
  104. }
  105. private func confirmation(_ ok: Bool) {
  106. WKInterfaceDevice.current().play(ok ? .success : .failure)
  107. withAnimation {
  108. confirmationSuccess = ok
  109. }
  110. DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
  111. withAnimation {
  112. self.isConfirmationViewActive = false
  113. }
  114. }
  115. }
  116. private func processState(_ state: WatchState) {
  117. glucose = state.glucose ?? "?"
  118. trend = state.trend ?? "?"
  119. delta = state.delta ?? "?"
  120. glucoseDate = state.glucoseDate
  121. lastLoopDate = state.lastLoopDate
  122. bolusIncrement = state.bolusIncrement
  123. maxCOB = state.maxCOB
  124. maxBolus = state.maxBolus
  125. bolusRecommended = state.bolusRecommended
  126. carbsRequired = state.carbsRequired
  127. iob = state.iob
  128. cob = state.cob
  129. tempTargets = state.tempTargets
  130. bolusAfterCarbs = state.bolusAfterCarbs ?? true
  131. eventualBG = state.eventualBG ?? ""
  132. lastUpdate = Date()
  133. }
  134. }
  135. extension WatchStateModel: WCSessionDelegate {
  136. func session(_: WCSession, activationDidCompleteWith state: WCSessionActivationState, error _: Error?) {
  137. print("WCSession activated: \(state == .activated)")
  138. requestState()
  139. }
  140. func session(_: WCSession, didReceiveMessage _: [String: Any]) {}
  141. func sessionReachabilityDidChange(_ session: WCSession) {
  142. print("WCSession Reachability: \(session.isReachable)")
  143. }
  144. func session(_: WCSession, didReceiveMessageData messageData: Data) {
  145. if let state = try? JSONDecoder().decode(WatchState.self, from: messageData) {
  146. DispatchQueue.main.async {
  147. // WKInterfaceDevice.current().play(.click)
  148. self.processState(state)
  149. }
  150. }
  151. }
  152. }