BolusStateModel.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. import CoreData
  2. import LoopKit
  3. import SwiftUI
  4. import Swinject
  5. extension Bolus {
  6. final class StateModel: BaseStateModel<Provider> {
  7. @Injected() var unlockmanager: UnlockManager!
  8. @Injected() var apsManager: APSManager!
  9. @Injected() var broadcaster: Broadcaster!
  10. @Injected() var pumpHistoryStorage: PumpHistoryStorage!
  11. // added for bolus calculator
  12. @Injected() var settings: SettingsManager!
  13. @Injected() var nsManager: NightscoutManager!
  14. let coredataContext = CoreDataStack.shared.persistentContainer.viewContext
  15. @Published var suggestion: Suggestion?
  16. @Published var amount: Decimal = 0
  17. @Published var insulinRecommended: Decimal = 0
  18. @Published var insulinRequired: Decimal = 0
  19. @Published var units: GlucoseUnits = .mmolL
  20. @Published var percentage: Decimal = 0
  21. @Published var threshold: Decimal = 0
  22. @Published var maxBolus: Decimal = 0
  23. @Published var errorString: Decimal = 0
  24. @Published var evBG: Int = 0
  25. @Published var insulin: Decimal = 0
  26. @Published var isf: Decimal = 0
  27. @Published var error: Bool = false
  28. @Published var minGuardBG: Decimal = 0
  29. @Published var minDelta: Decimal = 0
  30. @Published var expectedDelta: Decimal = 0
  31. @Published var minPredBG: Decimal = 0
  32. @Published var waitForSuggestion: Bool = false
  33. @Published var carbRatio: Decimal = 0
  34. var waitForSuggestionInitial: Bool = false
  35. // added for bolus calculator
  36. @Published var recentGlucose: BloodGlucose?
  37. @Published var target: Decimal = 0
  38. @Published var cob: Decimal = 0
  39. @Published var iob: Decimal = 0
  40. @Published var currentBG: Decimal = 0
  41. @Published var fifteenMinInsulin: Decimal = 0
  42. @Published var deltaBG: Decimal = 0
  43. @Published var targetDifferenceInsulin: Decimal = 0
  44. @Published var wholeCobInsulin: Decimal = 0
  45. @Published var iobInsulinReduction: Decimal = 0
  46. @Published var wholeCalc: Decimal = 0
  47. @Published var roundedWholeCalc: Decimal = 0
  48. @Published var insulinCalculated: Decimal = 0
  49. @Published var roundedInsulinCalculated: Decimal = 0
  50. @Published var fraction: Decimal = 0
  51. @Published var useCalc: Bool = false
  52. @Published var basal: Decimal = 0
  53. @Published var fattyMeals: Bool = false
  54. @Published var fattyMealFactor: Decimal = 0
  55. @Published var useFattyMealCorrectionFactor: Bool = false
  56. @Published var eventualBG: Int = 0
  57. @Published var meal: [CarbsEntry]?
  58. @Published var carbs: Decimal = 0
  59. @Published var fat: Decimal = 0
  60. @Published var protein: Decimal = 0
  61. @Published var note: String = ""
  62. override func subscribe() {
  63. setupInsulinRequired()
  64. broadcaster.register(SuggestionObserver.self, observer: self)
  65. units = settingsManager.settings.units
  66. percentage = settingsManager.settings.insulinReqPercentage
  67. threshold = provider.suggestion?.threshold ?? 0
  68. maxBolus = provider.pumpSettings().maxBolus
  69. // added
  70. fraction = settings.settings.overrideFactor
  71. useCalc = settings.settings.useCalc
  72. fattyMeals = settings.settings.fattyMeals
  73. fattyMealFactor = settings.settings.fattyMealFactor
  74. if waitForSuggestionInitial {
  75. apsManager.determineBasal()
  76. .receive(on: DispatchQueue.main)
  77. .sink { [weak self] ok in
  78. guard let self = self else { return }
  79. if !ok {
  80. self.waitForSuggestion = false
  81. self.insulinRequired = 0
  82. self.insulinRecommended = 0
  83. }
  84. }.store(in: &lifetime)
  85. }
  86. }
  87. func getDeltaBG() {
  88. let glucose = provider.fetchGlucose()
  89. guard glucose.count >= 3 else { return }
  90. let lastGlucose = glucose.last?.glucose ?? 0
  91. let thirdLastGlucose = glucose[glucose.count - 3]
  92. let delta = Decimal(lastGlucose) - Decimal(thirdLastGlucose.glucose)
  93. deltaBG = delta
  94. }
  95. // CALCULATIONS FOR THE BOLUS CALCULATOR
  96. func calculateInsulin() -> Decimal {
  97. // for mmol conversion
  98. var conversion: Decimal = 1.0
  99. if units == .mmolL {
  100. conversion = 0.0555
  101. }
  102. // insulin needed for the current blood glucose
  103. let targetDifference = (currentBG - target) * conversion
  104. targetDifferenceInsulin = targetDifference / isf
  105. // more or less insulin because of bg trend in the last 15 minutes
  106. fifteenMinInsulin = (deltaBG * conversion) / isf
  107. // determine whole COB for which we want to dose insulin for and then determine insulin for wholeCOB
  108. wholeCobInsulin = cob / carbRatio
  109. // determine how much the calculator reduces/ increases the bolus because of IOB
  110. iobInsulinReduction = (-1) * iob
  111. // adding everything together
  112. // add a calc for the case that no fifteenMinInsulin is available
  113. if deltaBG != 0 {
  114. wholeCalc = (targetDifferenceInsulin + iobInsulinReduction + wholeCobInsulin + fifteenMinInsulin)
  115. } else {
  116. // add (rare) case that no glucose value is available -> maybe display warning?
  117. // if no bg is available, ?? sets its value to 0
  118. if currentBG == 0 {
  119. wholeCalc = (iobInsulinReduction + wholeCobInsulin)
  120. } else {
  121. wholeCalc = (targetDifferenceInsulin + iobInsulinReduction + wholeCobInsulin)
  122. }
  123. }
  124. // rounding
  125. let wholeCalcAsDouble = Double(wholeCalc)
  126. roundedWholeCalc = Decimal(round(100 * wholeCalcAsDouble) / 100)
  127. // apply custom factor at the end of the calculations
  128. let result = wholeCalc * fraction
  129. // apply custom factor if fatty meal toggle in bolus calc config settings is on and the box for fatty meals is checked (in RootView)
  130. if useFattyMealCorrectionFactor {
  131. insulinCalculated = result * fattyMealFactor
  132. } else {
  133. insulinCalculated = result
  134. }
  135. // display no negative insulinCalculated
  136. insulinCalculated = max(insulinCalculated, 0)
  137. let insulinCalculatedAsDouble = Double(insulinCalculated)
  138. roundedInsulinCalculated = Decimal(round(100 * insulinCalculatedAsDouble) / 100)
  139. insulinCalculated = min(insulinCalculated, maxBolus)
  140. return apsManager
  141. .roundBolus(amount: max(insulinCalculated, 0))
  142. }
  143. func add() {
  144. guard amount > 0 else {
  145. showModal(for: nil)
  146. return
  147. }
  148. let maxAmount = Double(min(amount, provider.pumpSettings().maxBolus))
  149. unlockmanager.unlock()
  150. .sink { _ in } receiveValue: { [weak self] _ in
  151. guard let self = self else { return }
  152. self.apsManager.enactBolus(amount: maxAmount, isSMB: false)
  153. self.showModal(for: nil)
  154. }
  155. .store(in: &lifetime)
  156. }
  157. func setupInsulinRequired() {
  158. DispatchQueue.main.async {
  159. self.insulinRequired = self.provider.suggestion?.insulinReq ?? 0
  160. var conversion: Decimal = 1.0
  161. if self.units == .mmolL {
  162. conversion = 0.0555
  163. }
  164. self.evBG = self.provider.suggestion?.eventualBG ?? 0
  165. self.insulin = self.provider.suggestion?.insulinForManualBolus ?? 0
  166. self.target = self.provider.suggestion?.current_target ?? 0
  167. self.isf = self.provider.suggestion?.isf ?? 0
  168. self.iob = self.provider.suggestion?.iob ?? 0
  169. self.currentBG = (self.provider.suggestion?.bg ?? 0)
  170. self.cob = self.provider.suggestion?.cob ?? 0
  171. self.basal = self.provider.suggestion?.rate ?? 0
  172. self.carbRatio = self.provider.suggestion?.carbRatio ?? 0
  173. if self.settingsManager.settings.insulinReqPercentage != 100 {
  174. self.insulinRecommended = self.insulin * (self.settingsManager.settings.insulinReqPercentage / 100)
  175. } else { self.insulinRecommended = self.insulin }
  176. self.errorString = self.provider.suggestion?.manualBolusErrorString ?? 0
  177. if self.errorString != 0 {
  178. self.error = true
  179. self.minGuardBG = (self.provider.suggestion?.minGuardBG ?? 0) * conversion
  180. self.minDelta = (self.provider.suggestion?.minDelta ?? 0) * conversion
  181. self.expectedDelta = (self.provider.suggestion?.expectedDelta ?? 0) * conversion
  182. self.minPredBG = (self.provider.suggestion?.minPredBG ?? 0) * conversion
  183. } else { self.error = false }
  184. self.insulinRecommended = self.apsManager
  185. .roundBolus(amount: max(self.insulinRecommended, 0))
  186. if self.useCalc {
  187. self.getDeltaBG()
  188. self.insulinCalculated = self.calculateInsulin()
  189. }
  190. }
  191. }
  192. func backToCarbsView(complexEntry: Bool, _ id: String) {
  193. delete(deleteTwice: complexEntry, id: id)
  194. showModal(for: .addCarbs(editMode: complexEntry))
  195. }
  196. func delete(deleteTwice: Bool, id: String) {
  197. if deleteTwice {
  198. // DispatchQueue.safeMainSync {
  199. nsManager.deleteCarbs(
  200. at: id, isFPU: nil, fpuID: nil, syncID: id
  201. )
  202. nsManager.deleteCarbs(
  203. at: id + ".fpu", isFPU: nil, fpuID: nil, syncID: id
  204. )
  205. // }
  206. } else {
  207. nsManager.deleteCarbs(
  208. at: id, isFPU: nil, fpuID: nil, syncID: id
  209. )
  210. }
  211. }
  212. }
  213. }
  214. extension Bolus.StateModel: SuggestionObserver {
  215. func suggestionDidUpdate(_: Suggestion) {
  216. DispatchQueue.main.async {
  217. self.waitForSuggestion = false
  218. }
  219. setupInsulinRequired()
  220. }
  221. }