HomeStateModel.swift 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  1. import CGMBLEKitUI
  2. import Combine
  3. import CoreData
  4. import Foundation
  5. import LoopKitUI
  6. import Observation
  7. import SwiftDate
  8. import SwiftUI
  9. extension Home {
  10. @Observable final class StateModel: BaseStateModel<Provider> {
  11. @ObservationIgnored @Injected() var broadcaster: Broadcaster!
  12. @ObservationIgnored @Injected() var apsManager: APSManager!
  13. @ObservationIgnored @Injected() var pluginCGMManager: PluginManager!
  14. @ObservationIgnored @Injected() var fetchGlucoseManager: FetchGlucoseManager!
  15. @ObservationIgnored @Injected() var nightscoutManager: NightscoutManager!
  16. @ObservationIgnored @Injected() var determinationStorage: DeterminationStorage!
  17. @ObservationIgnored @Injected() var glucoseStorage: GlucoseStorage!
  18. @ObservationIgnored @Injected() var carbsStorage: CarbsStorage!
  19. @ObservationIgnored @Injected() var tempTargetStorage: TempTargetsStorage!
  20. @ObservationIgnored @Injected() var overrideStorage: OverrideStorage!
  21. @ObservationIgnored @Injected() var bluetoothManager: BluetoothStateManager!
  22. @ObservationIgnored @Injected() var iobService: IOBService!
  23. var cgmStateModel: CGMSettings.StateModel {
  24. CGMSettings.StateModel.shared
  25. }
  26. private let timer = DispatchTimer(timeInterval: 5)
  27. private(set) var filteredHours = 24
  28. var startMarker = Date(timeIntervalSinceNow: TimeInterval(hours: -24))
  29. var endMarker = Date(timeIntervalSinceNow: TimeInterval(hours: 3))
  30. var manualGlucose: [BloodGlucose] = []
  31. var uploadStats = false
  32. var recentGlucose: BloodGlucose?
  33. var maxBasal: Decimal = 2
  34. var basalProfile: [BasalProfileEntry] = []
  35. var bgTargets = BGTargets(from: OpenAPS.defaults(for: OpenAPS.Settings.bgTargets))
  36. ?? BGTargets(units: .mgdL, userPreferredUnits: .mgdL, targets: [])
  37. var targetProfiles: [TargetProfile] = []
  38. var timerDate = Date()
  39. var closedLoop = false
  40. var isLooping = false
  41. var statusTitle = ""
  42. var lastLoopDate: Date = .distantPast
  43. var battery: Battery?
  44. var reservoir: Decimal?
  45. var pumpName = ""
  46. var pumpExpiresAtDate: Date?
  47. var highTTraisesSens: Bool = false
  48. var lowTTlowersSens: Bool = false
  49. var isExerciseModeActive: Bool = false
  50. var settingHalfBasalTarget: Decimal = 160
  51. var percentage: Int = 100
  52. var shouldDisplayPumpSetupSheet = false
  53. var shouldDisplayCGMSetupSheet = false
  54. var errorMessage: String?
  55. var errorDate: Date?
  56. var bolusProgress: Decimal?
  57. var eventualBG: Int?
  58. var allowManualTemp = false
  59. var units: GlucoseUnits = .mgdL
  60. var pumpDisplayState: PumpDisplayState?
  61. var alarm: GlucoseAlarm?
  62. var manualTempBasal = false
  63. var isSmoothingEnabled = false
  64. var maxIOB: Decimal = 0.0
  65. var currentIOB: Decimal = 0.0
  66. var autosensMax: Decimal = 1.2
  67. var lowGlucose: Decimal = 70
  68. var highGlucose: Decimal = 180
  69. var currentGlucoseTarget: Decimal = 100
  70. var glucoseColorScheme: GlucoseColorScheme = .staticColor
  71. var eA1cDisplayUnit: EstimatedA1cDisplayUnit = .percent
  72. var displayXgridLines: Bool = false
  73. var displayYgridLines: Bool = false
  74. var thresholdLines: Bool = false
  75. var hours: Int16 = 6
  76. var totalBolus: Decimal = 0
  77. var isLoopStatusPresented: Bool = false
  78. var isLegendPresented: Bool = false
  79. var roundedTotalBolus: String = ""
  80. var selectedTab: Int = 0
  81. var waitForSuggestion: Bool = false
  82. var glucoseFromPersistence: [GlucoseStored] = []
  83. var latestTwoGlucoseValues: [GlucoseStored] = []
  84. var carbsFromPersistence: [CarbEntryStored] = []
  85. var fpusFromPersistence: [CarbEntryStored] = []
  86. var determinationsFromPersistence: [OrefDetermination] = []
  87. var enactedAndNonEnactedDeterminations: [OrefDetermination] = []
  88. var fetchedTDDs: [TDD] = []
  89. var insulinFromPersistence: [PumpEventStored] = []
  90. var tempBasals: [PumpEventStored] = []
  91. var suspensions: [PumpEventStored] = [] // all of the pump suspend and resume events
  92. var batteryFromPersistence: [OpenAPS_Battery] = []
  93. var lastPumpBolus: PumpEventStored?
  94. var overrides: [OverrideStored] = []
  95. var overrideRunStored: [OverrideRunStored] = []
  96. var tempTargetStored: [TempTargetStored] = []
  97. var tempTargetRunStored: [TempTargetRunStored] = []
  98. var isOverrideCancelled: Bool = false
  99. var preprocessedData: [(id: UUID, forecast: Forecast, forecastValue: ForecastValue)] = []
  100. var pumpStatusHighlightMessage: String?
  101. var pumpStatusBadgeImage: UIImage?
  102. var pumpStatusBadgeColor: Color?
  103. var cgmAvailable: Bool = false
  104. var listOfCGM: [CGMModel] = []
  105. var cgmCurrent = cgmDefaultModel
  106. var shouldRunDeleteOnSettingsChange = true
  107. var showCarbsRequiredBadge: Bool = true
  108. private(set) var setupPumpType: PumpConfig.PumpType = .minimed
  109. var minForecast: [Int] = []
  110. var maxForecast: [Int] = []
  111. var minCount: Int = 12 // count of Forecasts drawn in 5 min distances, i.e. 12 means a min of 1 hour
  112. var forecastDisplayType: ForecastDisplayType = .cone
  113. var minYAxisValue: Decimal = 39
  114. var maxYAxisValue: Decimal = 200
  115. var minValueCobChart: Decimal = 0
  116. var maxValueCobChart: Decimal = 20
  117. var minValueIobChart: Decimal = 0
  118. var maxValueIobChart: Decimal = 5
  119. let taskContext = CoreDataStack.shared.newTaskContext()
  120. let glucoseFetchContext = CoreDataStack.shared.newTaskContext()
  121. let carbsFetchContext = CoreDataStack.shared.newTaskContext()
  122. let fpuFetchContext = CoreDataStack.shared.newTaskContext()
  123. let determinationFetchContext = CoreDataStack.shared.newTaskContext()
  124. let tddFetchContext = CoreDataStack.shared.newTaskContext()
  125. let pumpHistoryFetchContext = CoreDataStack.shared.newTaskContext()
  126. let overrideFetchContext = CoreDataStack.shared.newTaskContext()
  127. let tempTargetFetchContext = CoreDataStack.shared.newTaskContext()
  128. let batteryFetchContext = CoreDataStack.shared.newTaskContext()
  129. let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  130. // Queue for handling Core Data change notifications
  131. private let queue = DispatchQueue(label: "HomeStateModel.queue", qos: .userInitiated)
  132. private var coreDataPublisher: AnyPublisher<Set<NSManagedObjectID>, Never>?
  133. private var subscriptions = Set<AnyCancellable>()
  134. typealias PumpEvent = PumpEventStored.EventType
  135. override init() {
  136. super.init()
  137. }
  138. override func subscribe() {
  139. coreDataPublisher =
  140. changedObjectsOnManagedObjectContextDidSavePublisher()
  141. .receive(on: queue)
  142. .share()
  143. .eraseToAnyPublisher()
  144. registerSubscribers()
  145. registerHandlers()
  146. // Parallelize Setup functions
  147. setupHomeViewConcurrently()
  148. }
  149. private func setupHomeViewConcurrently() {
  150. Task {
  151. // We need to initialize settings and observers first
  152. await self.setupSettings()
  153. await self.setupPumpSettings()
  154. await self.setupCGMSettings()
  155. self.registerObservers()
  156. // The rest can be initialized concurrently
  157. await withTaskGroup(of: Void.self) { group in
  158. group.addTask {
  159. self.setupGlucoseArray()
  160. }
  161. group.addTask {
  162. self.setupCarbsArray()
  163. }
  164. group.addTask {
  165. self.setupFPUsArray()
  166. }
  167. group.addTask {
  168. self.setupDeterminationsArray()
  169. }
  170. group.addTask {
  171. self.setupTDDArray()
  172. }
  173. group.addTask {
  174. self.setupInsulinArray()
  175. }
  176. group.addTask {
  177. self.setupLastBolus()
  178. }
  179. group.addTask {
  180. self.setupBatteryArray()
  181. }
  182. group.addTask {
  183. await self.setupBasalProfile()
  184. }
  185. group.addTask {
  186. await self.setupGlucoseTargets()
  187. }
  188. group.addTask {
  189. self.setupReservoir()
  190. }
  191. group.addTask {
  192. self.setupOverrides()
  193. }
  194. group.addTask {
  195. self.setupOverrideRunStored()
  196. }
  197. group.addTask {
  198. self.setupTempTargetsStored()
  199. }
  200. group.addTask {
  201. self.setupTempTargetsRunStored()
  202. }
  203. group.addTask {
  204. self.iobService.updateIOB()
  205. }
  206. }
  207. }
  208. }
  209. // These combine subscribers are only necessary due to the batch inserts of glucose/FPUs which do not trigger a ManagedObjectContext change notification
  210. private func registerSubscribers() {
  211. iobService.iobPublisher
  212. .receive(on: DispatchQueue.main)
  213. .sink { [weak self] _ in
  214. guard let self = self else { return }
  215. self.currentIOB = self.iobService.currentIOB ?? 0
  216. }
  217. .store(in: &subscriptions)
  218. glucoseStorage.updatePublisher
  219. .receive(on: queue)
  220. .sink { [weak self] _ in
  221. guard let self = self else { return }
  222. self.setupGlucoseArray()
  223. }
  224. .store(in: &subscriptions)
  225. carbsStorage.updatePublisher
  226. .receive(on: queue)
  227. .sink { [weak self] _ in
  228. guard let self = self else { return }
  229. self.setupFPUsArray()
  230. }
  231. .store(in: &subscriptions)
  232. }
  233. private func registerHandlers() {
  234. coreDataPublisher?.filteredByEntityName("OrefDetermination").sink { [weak self] _ in
  235. guard let self = self else { return }
  236. self.setupDeterminationsArray()
  237. }.store(in: &subscriptions)
  238. coreDataPublisher?.filteredByEntityName("TDDStored").sink { [weak self] _ in
  239. guard let self = self else { return }
  240. self.setupTDDArray()
  241. }.store(in: &subscriptions)
  242. coreDataPublisher?.filteredByEntityName("GlucoseStored").sink { [weak self] _ in
  243. guard let self = self else { return }
  244. self.setupGlucoseArray()
  245. }.store(in: &subscriptions)
  246. coreDataPublisher?.filteredByEntityName("CarbEntryStored").sink { [weak self] _ in
  247. guard let self = self else { return }
  248. self.setupCarbsArray()
  249. }.store(in: &subscriptions)
  250. coreDataPublisher?.filteredByEntityName("PumpEventStored").sink { [weak self] _ in
  251. guard let self = self else { return }
  252. self.setupInsulinArray()
  253. self.setupLastBolus()
  254. self.displayPumpStatusHighlightMessage()
  255. self.displayPumpStatusBadge()
  256. }.store(in: &subscriptions)
  257. coreDataPublisher?.filteredByEntityName("OpenAPS_Battery").sink { [weak self] _ in
  258. guard let self = self else { return }
  259. self.setupBatteryArray()
  260. }.store(in: &subscriptions)
  261. coreDataPublisher?.filteredByEntityName("OverrideStored").sink { [weak self] _ in
  262. guard let self = self else { return }
  263. self.setupOverrides()
  264. }.store(in: &subscriptions)
  265. coreDataPublisher?.filteredByEntityName("OverrideRunStored").sink { [weak self] _ in
  266. guard let self = self else { return }
  267. self.setupOverrideRunStored()
  268. }.store(in: &subscriptions)
  269. coreDataPublisher?.filteredByEntityName("TempTargetStored").sink { [weak self] _ in
  270. guard let self = self else { return }
  271. self.setupTempTargetsStored()
  272. }.store(in: &subscriptions)
  273. coreDataPublisher?.filteredByEntityName("TempTargetRunStored").sink { [weak self] _ in
  274. guard let self = self else { return }
  275. self.setupTempTargetsRunStored()
  276. }.store(in: &subscriptions)
  277. }
  278. private func registerObservers() {
  279. broadcaster.register(DeterminationObserver.self, observer: self)
  280. broadcaster.register(SettingsObserver.self, observer: self)
  281. broadcaster.register(PreferencesObserver.self, observer: self)
  282. broadcaster.register(PumpSettingsObserver.self, observer: self)
  283. broadcaster.register(BasalProfileObserver.self, observer: self)
  284. broadcaster.register(BGTargetsObserver.self, observer: self)
  285. broadcaster.register(PumpReservoirObserver.self, observer: self)
  286. broadcaster.register(PumpDeactivatedObserver.self, observer: self)
  287. timer.eventHandler = {
  288. DispatchQueue.main.async { [weak self] in
  289. self?.timerDate = Date()
  290. }
  291. }
  292. timer.resume()
  293. apsManager.isLooping
  294. .receive(on: DispatchQueue.main)
  295. .weakAssign(to: \.isLooping, on: self)
  296. .store(in: &lifetime)
  297. apsManager.lastLoopDateSubject
  298. .receive(on: DispatchQueue.main)
  299. .weakAssign(to: \.lastLoopDate, on: self)
  300. .store(in: &lifetime)
  301. apsManager.pumpName
  302. .receive(on: DispatchQueue.main)
  303. .weakAssign(to: \.pumpName, on: self)
  304. .store(in: &lifetime)
  305. apsManager.pumpExpiresAtDate
  306. .receive(on: DispatchQueue.main)
  307. .weakAssign(to: \.pumpExpiresAtDate, on: self)
  308. .store(in: &lifetime)
  309. apsManager.lastError
  310. .receive(on: DispatchQueue.main)
  311. .map { [weak self] error in
  312. self?.errorDate = error == nil ? nil : Date()
  313. if let error = error {
  314. info(.default, String(describing: error), notificationText: error.localizedDescription)
  315. }
  316. return error?.localizedDescription
  317. }
  318. .weakAssign(to: \.errorMessage, on: self)
  319. .store(in: &lifetime)
  320. apsManager.bolusProgress
  321. .receive(on: DispatchQueue.main)
  322. .weakAssign(to: \.bolusProgress, on: self)
  323. .store(in: &lifetime)
  324. apsManager.pumpDisplayState
  325. .receive(on: DispatchQueue.main)
  326. .sink { [weak self] state in
  327. guard let self = self else { return }
  328. self.pumpDisplayState = state
  329. if state == nil {
  330. self.reservoir = nil
  331. self.battery = nil
  332. self.pumpName = ""
  333. self.pumpExpiresAtDate = nil
  334. self.shouldDisplayPumpSetupSheet = false
  335. } else {
  336. self.setupReservoir()
  337. self.displayPumpStatusHighlightMessage()
  338. self.displayPumpStatusBadge()
  339. self.setupBatteryArray()
  340. }
  341. }
  342. .store(in: &lifetime)
  343. }
  344. private enum SettingType {
  345. case basal
  346. case carbRatio
  347. case bgTarget
  348. case isf
  349. }
  350. @MainActor private func setupSettings() async {
  351. units = settingsManager.settings.units
  352. allowManualTemp = !settingsManager.settings.closedLoop
  353. closedLoop = settingsManager.settings.closedLoop
  354. lastLoopDate = apsManager.lastLoopDate
  355. alarm = provider.glucoseStorage.alarm
  356. manualTempBasal = apsManager.isManualTempBasal
  357. isSmoothingEnabled = settingsManager.settings.smoothGlucose
  358. glucoseColorScheme = settingsManager.settings.glucoseColorScheme
  359. autosensMax = settingsManager.preferences.autosensMax
  360. lowGlucose = settingsManager.settings.low
  361. highGlucose = settingsManager.settings.high
  362. eA1cDisplayUnit = settingsManager.settings.eA1cDisplayUnit
  363. displayXgridLines = settingsManager.settings.xGridLines
  364. displayYgridLines = settingsManager.settings.yGridLines
  365. thresholdLines = settingsManager.settings.rulerMarks
  366. showCarbsRequiredBadge = settingsManager.settings.showCarbsRequiredBadge
  367. forecastDisplayType = settingsManager.settings.forecastDisplayType
  368. isExerciseModeActive = settingsManager.preferences.exerciseMode
  369. highTTraisesSens = settingsManager.preferences.highTemptargetRaisesSensitivity
  370. lowTTlowersSens = settingsManager.preferences.lowTemptargetLowersSensitivity
  371. settingHalfBasalTarget = settingsManager.preferences.halfBasalExerciseTarget
  372. maxIOB = settingsManager.preferences.maxIOB
  373. }
  374. @MainActor private func setupCGMSettings() async {
  375. cgmAvailable = fetchGlucoseManager.cgmGlucoseSourceType != CGMType.none
  376. listOfCGM = (
  377. CGMType.allCases.filter { $0 != CGMType.plugin }.map {
  378. CGMModel(id: $0.id, type: $0, displayName: $0.displayName, subtitle: $0.subtitle)
  379. } +
  380. pluginCGMManager.availableCGMManagers.map {
  381. CGMModel(
  382. id: $0.identifier,
  383. type: CGMType.plugin,
  384. displayName: $0.localizedTitle,
  385. subtitle: $0.localizedTitle
  386. )
  387. }
  388. ).sorted(by: { lhs, rhs in
  389. if lhs.displayName == "None" {
  390. return true
  391. } else if rhs.displayName == "None" {
  392. return false
  393. } else {
  394. return lhs.displayName < rhs.displayName
  395. }
  396. })
  397. switch settingsManager.settings.cgm {
  398. case .plugin:
  399. if let cgmPluginInfo = listOfCGM.first(where: { $0.id == settingsManager.settings.cgmPluginIdentifier }) {
  400. cgmCurrent = CGMModel(
  401. id: settingsManager.settings.cgmPluginIdentifier,
  402. type: .plugin,
  403. displayName: cgmPluginInfo.displayName,
  404. subtitle: cgmPluginInfo.subtitle
  405. )
  406. } else {
  407. // no more type of plugin available - fallback to default
  408. cgmCurrent = cgmDefaultModel
  409. }
  410. default:
  411. cgmCurrent = CGMModel(
  412. id: settingsManager.settings.cgm.id,
  413. type: settingsManager.settings.cgm,
  414. displayName: settingsManager.settings.cgm.displayName,
  415. subtitle: settingsManager.settings.cgm.subtitle
  416. )
  417. }
  418. }
  419. func addPump(_ type: PumpConfig.PumpType) {
  420. setupPumpType = type
  421. shouldDisplayPumpSetupSheet = true
  422. }
  423. func addCGM(cgm: CGMModel) {
  424. cgmCurrent = cgm
  425. switch cgmCurrent.type {
  426. case .plugin:
  427. shouldDisplayCGMSetupSheet = true
  428. default:
  429. shouldDisplayCGMSetupSheet = true
  430. settingsManager.settings.cgm = cgmCurrent.type
  431. settingsManager.settings.cgmPluginIdentifier = ""
  432. fetchGlucoseManager.updateGlucoseSource(cgmGlucoseSourceType: cgmCurrent.type, cgmGlucosePluginId: cgmCurrent.id)
  433. broadcaster.notify(GlucoseObserver.self, on: .main) {
  434. $0.glucoseDidUpdate([])
  435. }
  436. }
  437. }
  438. func deleteCGM() {
  439. fetchGlucoseManager.performOnCGMManagerQueue {
  440. // Call plugin functionality on the manager queue (or at least attempt to)
  441. Task {
  442. await self.fetchGlucoseManager?.deleteGlucoseSource()
  443. // UI updates go back to Main
  444. await MainActor.run {
  445. self.shouldDisplayCGMSetupSheet = false
  446. self.broadcaster.notify(GlucoseObserver.self, on: .main) {
  447. $0.glucoseDidUpdate([])
  448. }
  449. }
  450. }
  451. }
  452. }
  453. /// Display the eventual status message provided by the manager of the pump
  454. /// Only display if state is warning or critical message else return nil
  455. private func displayPumpStatusHighlightMessage(_ didDeactivate: Bool = false) {
  456. DispatchQueue.main.async { [weak self] in
  457. guard let self = self else { return }
  458. if let statusHighlight = self.provider.deviceManager.pumpManager?.pumpStatusHighlight,
  459. statusHighlight.state == .warning || statusHighlight.state == .critical, !didDeactivate
  460. {
  461. pumpStatusHighlightMessage = (statusHighlight.state == .warning ? "⚠️\n" : "‼️\n") + statusHighlight
  462. .localizedMessage
  463. } else {
  464. pumpStatusHighlightMessage = nil
  465. }
  466. }
  467. }
  468. private func displayPumpStatusBadge(_ didDeactivate: Bool = false) {
  469. DispatchQueue.main.async { [weak self] in
  470. guard let self = self else { return }
  471. if let statusBadge = self.provider.deviceManager.pumpManager?.pumpStatusBadge,
  472. let image = statusBadge.image, !didDeactivate
  473. {
  474. pumpStatusBadgeImage = image
  475. pumpStatusBadgeColor = statusBadge.state == .critical ? .critical : .warning
  476. } else {
  477. pumpStatusBadgeImage = nil
  478. pumpStatusBadgeColor = nil
  479. }
  480. }
  481. }
  482. func runLoop() {
  483. provider.heartbeatNow()
  484. }
  485. func showProgressView() {
  486. glucoseStorage
  487. .isGlucoseDataFresh(glucoseFromPersistence.first?.date) ? (waitForSuggestion = true) : (waitForSuggestion = false)
  488. }
  489. func cancelBolus() {
  490. Task {
  491. await apsManager.cancelBolus(nil)
  492. // perform determine basal sync, otherwise you have could end up with too much iob when opening the calculator again
  493. try await apsManager.determineBasalSync()
  494. }
  495. }
  496. private func setupPumpSettings() async {
  497. let maxBasal = await provider.pumpSettings().maxBasal
  498. await MainActor.run {
  499. self.maxBasal = maxBasal
  500. }
  501. }
  502. private func setupBasalProfile() async {
  503. let basalProfile = await provider.getBasalProfile()
  504. await MainActor.run {
  505. self.basalProfile = basalProfile
  506. }
  507. }
  508. private func setupGlucoseTargets() async {
  509. let bgTargets = await provider.getBGTargets()
  510. let targetProfiles = processFetchedTargets(bgTargets, startMarker: startMarker)
  511. await MainActor.run {
  512. self.bgTargets = bgTargets
  513. self.targetProfiles = targetProfiles
  514. }
  515. }
  516. private func setupReservoir() {
  517. Task {
  518. let reservoir = await provider.pumpReservoir()
  519. await MainActor.run {
  520. self.reservoir = reservoir
  521. }
  522. }
  523. }
  524. private func getCurrentGlucoseTarget() async {
  525. let now = Date()
  526. let calendar = Calendar.current
  527. let entries: [(start: String, value: Decimal)] = bgTargets.targets.map { ($0.start, $0.low) }
  528. for (index, entry) in entries.enumerated() {
  529. guard let entryTime = TherapySettingsUtil.parseTime(entry.start) else {
  530. debug(.default, "Invalid entry start time: \(entry.start)")
  531. continue
  532. }
  533. let entryComponents = calendar.dateComponents([.hour, .minute, .second], from: entryTime)
  534. let entryStartTime = calendar.date(
  535. bySettingHour: entryComponents.hour!,
  536. minute: entryComponents.minute!,
  537. second: entryComponents.second!,
  538. of: now
  539. )!
  540. let entryEndTime: Date
  541. if index < entries.count - 1,
  542. let nextEntryTime = TherapySettingsUtil.parseTime(entries[index + 1].start)
  543. {
  544. let nextEntryComponents = calendar.dateComponents([.hour, .minute, .second], from: nextEntryTime)
  545. entryEndTime = calendar.date(
  546. bySettingHour: nextEntryComponents.hour!,
  547. minute: nextEntryComponents.minute!,
  548. second: nextEntryComponents.second!,
  549. of: now
  550. )!
  551. } else {
  552. entryEndTime = calendar.date(byAdding: .day, value: 1, to: entryStartTime)!
  553. }
  554. if now >= entryStartTime, now < entryEndTime {
  555. await MainActor.run {
  556. currentGlucoseTarget = entry.value
  557. }
  558. return
  559. }
  560. }
  561. }
  562. }
  563. }
  564. extension Home.StateModel:
  565. DeterminationObserver,
  566. SettingsObserver,
  567. PreferencesObserver,
  568. PumpSettingsObserver,
  569. BasalProfileObserver,
  570. BGTargetsObserver,
  571. PumpReservoirObserver,
  572. PumpDeactivatedObserver
  573. {
  574. func determinationDidUpdate(_: Determination) {
  575. waitForSuggestion = false
  576. }
  577. func settingsDidChange(_ settings: TrioSettings) {
  578. allowManualTemp = !settings.closedLoop
  579. closedLoop = settingsManager.settings.closedLoop
  580. units = settingsManager.settings.units
  581. manualTempBasal = apsManager.isManualTempBasal
  582. isSmoothingEnabled = settingsManager.settings.smoothGlucose
  583. lowGlucose = settingsManager.settings.low
  584. highGlucose = settingsManager.settings.high
  585. Task {
  586. await getCurrentGlucoseTarget()
  587. await setupGlucoseTargets()
  588. }
  589. eA1cDisplayUnit = settingsManager.settings.eA1cDisplayUnit
  590. glucoseColorScheme = settingsManager.settings.glucoseColorScheme
  591. displayXgridLines = settingsManager.settings.xGridLines
  592. displayYgridLines = settingsManager.settings.yGridLines
  593. thresholdLines = settingsManager.settings.rulerMarks
  594. showCarbsRequiredBadge = settingsManager.settings.showCarbsRequiredBadge
  595. forecastDisplayType = settingsManager.settings.forecastDisplayType
  596. cgmAvailable = (fetchGlucoseManager.cgmGlucoseSourceType != CGMType.none)
  597. displayPumpStatusHighlightMessage()
  598. displayPumpStatusBadge()
  599. setupBatteryArray()
  600. Task {
  601. await setupCGMSettings()
  602. }
  603. if settingsManager.settings.cgm == .none, shouldRunDeleteOnSettingsChange {
  604. shouldRunDeleteOnSettingsChange = false
  605. cgmCurrent = cgmDefaultModel
  606. DispatchQueue.main.async {
  607. self.broadcaster.notify(GlucoseObserver.self, on: .main) {
  608. $0.glucoseDidUpdate([])
  609. }
  610. }
  611. } else {
  612. shouldRunDeleteOnSettingsChange = true
  613. }
  614. }
  615. func preferencesDidChange(_: Preferences) {
  616. autosensMax = settingsManager.preferences.autosensMax
  617. settingHalfBasalTarget = settingsManager.preferences.halfBasalExerciseTarget
  618. highTTraisesSens = settingsManager.preferences.highTemptargetRaisesSensitivity
  619. isExerciseModeActive = settingsManager.preferences.exerciseMode
  620. lowTTlowersSens = settingsManager.preferences.lowTemptargetLowersSensitivity
  621. maxIOB = settingsManager.preferences.maxIOB
  622. }
  623. func pumpSettingsDidChange(_: PumpSettings) {
  624. Task {
  625. await setupPumpSettings()
  626. setupBatteryArray()
  627. }
  628. }
  629. func basalProfileDidChange(_: [BasalProfileEntry]) {
  630. Task {
  631. await setupBasalProfile()
  632. }
  633. }
  634. func bgTargetsDidChange(_: BGTargets) {
  635. Task {
  636. await setupGlucoseTargets()
  637. }
  638. }
  639. func pumpReservoirDidChange(_: Decimal) {
  640. setupReservoir()
  641. displayPumpStatusHighlightMessage()
  642. displayPumpStatusBadge()
  643. }
  644. func pumpDeactivatedDidChange() {
  645. displayPumpStatusHighlightMessage(true)
  646. displayPumpStatusBadge(true)
  647. batteryFromPersistence = []
  648. }
  649. }
  650. extension Home.StateModel: PumpManagerOnboardingDelegate {
  651. func pumpManagerOnboarding(didCreatePumpManager pumpManager: PumpManagerUI) {
  652. provider.apsManager.pumpManager = pumpManager
  653. if let insulinType = pumpManager.status.insulinType {
  654. settingsManager.updateInsulinCurve(insulinType)
  655. }
  656. }
  657. func pumpManagerOnboarding(didOnboardPumpManager _: PumpManagerUI) {
  658. // nothing to do
  659. }
  660. func pumpManagerOnboarding(didPauseOnboarding _: PumpManagerUI) {
  661. // nothing to do
  662. }
  663. }