OverrideProfilesStateModel.swift 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966
  1. import CoreData
  2. import SwiftUI
  3. extension OverrideProfilesConfig {
  4. final class StateModel: BaseStateModel<Provider> {
  5. @Injected() var broadcaster: Broadcaster!
  6. @Injected() var storage: TempTargetsStorage!
  7. @Injected() var apsManager: APSManager!
  8. @Injected() var overrideStorage: OverrideStorage!
  9. @Published var overrideSliderPercentage: Double = 100
  10. @Published var isEnabled = false
  11. @Published var indefinite = true
  12. @Published var overrideDuration: Decimal = 0
  13. @Published var target: Decimal = 0
  14. @Published var shouldOverrideTarget: Bool = false
  15. @Published var smbIsOff: Bool = false
  16. @Published var id = ""
  17. @Published var overrideName: String = ""
  18. @Published var isPreset: Bool = false
  19. @Published var overridePresets: [OverrideStored] = []
  20. @Published var advancedSettings: Bool = false
  21. @Published var isfAndCr: Bool = true
  22. @Published var isf: Bool = true
  23. @Published var cr: Bool = true
  24. @Published var smbIsScheduledOff: Bool = false
  25. @Published var start: Decimal = 0
  26. @Published var end: Decimal = 23
  27. @Published var smbMinutes: Decimal = 0
  28. @Published var uamMinutes: Decimal = 0
  29. @Published var defaultSmbMinutes: Decimal = 0
  30. @Published var defaultUamMinutes: Decimal = 0
  31. <<<<<<< HEAD
  32. @Published var selectedTab: Tab = .overrides
  33. @Published var activeOverrideName: String = ""
  34. @Published var currentActiveOverride: OverrideStored?
  35. @Published var showOverrideEditSheet = false
  36. @Published var showInvalidTargetAlert = false
  37. =======
  38. >>>>>>> 9672da256c317a314acc76d6e4f6e82cc174d133
  39. var units: GlucoseUnits = .mgdL
  40. // temp target stuff
  41. @Published var low: Decimal = 0
  42. @Published var high: Decimal = 0
  43. @Published var durationTT: Decimal = 0
  44. @Published var date = Date()
  45. @Published var newPresetName = ""
  46. @Published var presetsTT: [TempTarget] = []
  47. @Published var percentageTT = 100.0
  48. @Published var maxValue: Decimal = 1.2
  49. @Published var viewPercantage = false
  50. @Published var hbt: Double = 160
  51. @Published var didSaveSettings: Bool = false
  52. var alertMessage: String {
  53. let target: String = units == .mgdL ? "70-270 mg/dl" : "4-15 mmol/l"
  54. return "Please enter a valid target between" + " \(target)."
  55. }
  56. private var formatter: NumberFormatter {
  57. let formatter = NumberFormatter()
  58. formatter.numberStyle = .decimal
  59. formatter.maximumFractionDigits = 0
  60. return formatter
  61. }
  62. private var glucoseFormatter: NumberFormatter {
  63. let formatter = NumberFormatter()
  64. formatter.numberStyle = .decimal
  65. formatter.maximumFractionDigits = 0
  66. if units == .mmolL {
  67. formatter.maximumFractionDigits = 1
  68. }
  69. formatter.roundingMode = .halfUp
  70. return formatter
  71. }
  72. override func subscribe() {
  73. setupNotification()
  74. units = settingsManager.settings.units
  75. defaultSmbMinutes = settingsManager.preferences.maxSMBBasalMinutes
  76. defaultUamMinutes = settingsManager.preferences.maxUAMSMBBasalMinutes
  77. <<<<<<< HEAD
  78. setupOverridePresetsArray()
  79. updateLatestOverrideConfiguration()
  80. presetsTT = storage.presets()
  81. maxValue = settingsManager.preferences.autosensMax
  82. broadcaster.register(SettingsObserver.self, observer: self)
  83. =======
  84. presets = [OverridePresets(context: coredataContext)]
  85. >>>>>>> 9672da256c317a314acc76d6e4f6e82cc174d133
  86. }
  87. let coredataContext = CoreDataStack.shared.newTaskContext()
  88. let viewContext = CoreDataStack.shared.persistentContainer.viewContext
  89. <<<<<<< HEAD
  90. func isInputInvalid(target: Decimal) -> Bool {
  91. guard target != 0 else { return false }
  92. if units == .mgdL,
  93. target < 70 || target > 270
  94. {
  95. showInvalidTargetAlert = true
  96. return true
  97. } else if units == .mmolL,
  98. target < 4 || target > 15
  99. {
  100. showInvalidTargetAlert = true
  101. return true
  102. } else {
  103. return false
  104. =======
  105. struct ProfileViewData {
  106. let target: Decimal
  107. let duration: Decimal
  108. let name: String
  109. let percent: Double
  110. let perpetual: Bool
  111. let durationString: String
  112. let scheduledSMBString: String
  113. let smbString: String
  114. let targetString: String
  115. let maxMinutesSMB: Decimal
  116. let maxMinutesUAM: Decimal
  117. let isfString: String
  118. let crString: String
  119. let isfAndCRString: String
  120. }
  121. func profileViewData(for preset: OverridePresets) -> ProfileViewData {
  122. let target = units == .mmolL ? (((preset.target ?? 0) as NSDecimalNumber) as Decimal)
  123. .asMmolL : (preset.target ?? 0) as Decimal
  124. let duration = (preset.duration ?? 0) as Decimal
  125. let name = ((preset.name ?? "") == "") || (preset.name?.isEmpty ?? true) ? "" : preset.name!
  126. let percent = preset.percentage / 100
  127. let perpetual = preset.indefinite
  128. let durationString = perpetual ? "" : "\(formatter.string(from: duration as NSNumber)!)"
  129. let scheduledSMBString = (preset.smbIsOff && preset.smbIsScheduledOff) ? "Scheduled SMBs" : ""
  130. let smbString = (preset.smbIsOff && scheduledSMBString == "") ? "SMBs are off" : ""
  131. let targetString = target != 0 ? "\(glucoseFormatter.string(from: target as NSNumber)!)" : ""
  132. let maxMinutesSMB = (preset.smbMinutes as Decimal?) != nil ? (preset.smbMinutes ?? 0) as Decimal : 0
  133. let maxMinutesUAM = (preset.uamMinutes as Decimal?) != nil ? (preset.uamMinutes ?? 0) as Decimal : 0
  134. let isfString = preset.isf ? "ISF" : ""
  135. let crString = preset.cr ? "CR" : ""
  136. let dash = crString != "" ? "/" : ""
  137. let isfAndCRString = isfString + dash + crString
  138. return ProfileViewData(
  139. target: target,
  140. duration: duration,
  141. name: name,
  142. percent: percent,
  143. perpetual: perpetual,
  144. durationString: durationString,
  145. scheduledSMBString: scheduledSMBString,
  146. smbString: smbString,
  147. targetString: targetString,
  148. maxMinutesSMB: maxMinutesSMB,
  149. maxMinutesUAM: maxMinutesUAM,
  150. isfString: isfString,
  151. crString: crString,
  152. isfAndCRString: isfAndCRString
  153. )
  154. }
  155. func saveSettings() {
  156. coredataContext.perform { [self] in
  157. let saveOverride = Override(context: self.coredataContext)
  158. saveOverride.duration = self.duration as NSDecimalNumber
  159. saveOverride.indefinite = self._indefinite
  160. saveOverride.percentage = self.percentage
  161. saveOverride.enabled = true
  162. saveOverride.smbIsOff = self.smbIsOff
  163. if self.isPreset {
  164. saveOverride.isPreset = true
  165. saveOverride.id = id
  166. } else { saveOverride.isPreset = false }
  167. saveOverride.date = Date()
  168. if override_target {
  169. if units == .mmolL {
  170. target = target.asMgdL
  171. }
  172. saveOverride.target = target as NSDecimalNumber
  173. } else { saveOverride.target = 0 }
  174. if advancedSettings {
  175. saveOverride.advancedSettings = true
  176. if !isfAndCr {
  177. saveOverride.isfAndCr = false
  178. saveOverride.isf = isf
  179. saveOverride.cr = cr
  180. } else { saveOverride.isfAndCr = true }
  181. if smbIsScheduledOff {
  182. saveOverride.smbIsScheduledOff = true
  183. saveOverride.start = start as NSDecimalNumber
  184. saveOverride.end = end as NSDecimalNumber
  185. } else { saveOverride.smbIsScheduledOff = false }
  186. saveOverride.smbMinutes = smbMinutes as NSDecimalNumber
  187. saveOverride.uamMinutes = uamMinutes as NSDecimalNumber
  188. }
  189. try? self.coredataContext.save()
  190. }
  191. }
  192. func savePreset() {
  193. coredataContext.perform { [self] in
  194. let saveOverride = OverridePresets(context: self.coredataContext)
  195. saveOverride.duration = self.duration as NSDecimalNumber
  196. saveOverride.indefinite = self._indefinite
  197. saveOverride.percentage = self.percentage
  198. saveOverride.smbIsOff = self.smbIsOff
  199. saveOverride.name = self.profileName
  200. self.profileName = ""
  201. id = UUID().uuidString
  202. self.isPreset.toggle()
  203. saveOverride.id = id
  204. saveOverride.date = Date()
  205. if override_target {
  206. saveOverride.target = (
  207. units == .mmolL
  208. ? target.asMgdL
  209. : target
  210. ) as NSDecimalNumber
  211. } else { saveOverride.target = 0 }
  212. if advancedSettings {
  213. saveOverride.advancedSettings = true
  214. if !isfAndCr {
  215. saveOverride.isfAndCr = false
  216. saveOverride.isf = isf
  217. saveOverride.cr = cr
  218. } else { saveOverride.isfAndCr = true }
  219. if smbIsScheduledOff {
  220. saveOverride.smbIsScheduledOff = true
  221. saveOverride.start = start as NSDecimalNumber
  222. saveOverride.end = end as NSDecimalNumber
  223. } else { smbIsScheduledOff = false }
  224. saveOverride.smbMinutes = smbMinutes as NSDecimalNumber
  225. saveOverride.uamMinutes = uamMinutes as NSDecimalNumber
  226. }
  227. try? self.coredataContext.save()
  228. }
  229. }
  230. func selectProfile(id_: String) {
  231. guard id_ != "" else { return }
  232. coredataContext.performAndWait {
  233. var profileArray = [OverridePresets]()
  234. let requestProfiles = OverridePresets.fetchRequest() as NSFetchRequest<OverridePresets>
  235. try? profileArray = coredataContext.fetch(requestProfiles)
  236. guard let profile = profileArray.filter({ $0.id == id_ }).first else { return }
  237. let saveOverride = Override(context: self.coredataContext)
  238. saveOverride.duration = (profile.duration ?? 0) as NSDecimalNumber
  239. saveOverride.indefinite = profile.indefinite
  240. saveOverride.percentage = profile.percentage
  241. saveOverride.enabled = true
  242. saveOverride.smbIsOff = profile.smbIsOff
  243. saveOverride.isPreset = true
  244. saveOverride.date = Date()
  245. saveOverride.target = profile.target
  246. saveOverride.id = id_
  247. if profile.advancedSettings {
  248. saveOverride.advancedSettings = true
  249. if !isfAndCr {
  250. saveOverride.isfAndCr = false
  251. saveOverride.isf = profile.isf
  252. saveOverride.cr = profile.cr
  253. } else { saveOverride.isfAndCr = true }
  254. if profile.smbIsScheduledOff {
  255. saveOverride.smbIsScheduledOff = true
  256. saveOverride.start = profile.start
  257. saveOverride.end = profile.end
  258. } else { saveOverride.smbIsScheduledOff = false }
  259. saveOverride.smbMinutes = (profile.smbMinutes ?? 0) as NSDecimalNumber
  260. saveOverride.uamMinutes = (profile.uamMinutes ?? 0) as NSDecimalNumber
  261. }
  262. try? self.coredataContext.save()
  263. }
  264. }
  265. func savedSettings() {
  266. coredataContext.performAndWait {
  267. var overrideArray = [Override]()
  268. let requestEnabled = Override.fetchRequest() as NSFetchRequest<Override>
  269. let sortIsEnabled = NSSortDescriptor(key: "date", ascending: false)
  270. requestEnabled.sortDescriptors = [sortIsEnabled]
  271. try? overrideArray = coredataContext.fetch(requestEnabled)
  272. isEnabled = overrideArray.first?.enabled ?? false
  273. percentage = overrideArray.first?.percentage ?? 100
  274. _indefinite = overrideArray.first?.indefinite ?? true
  275. duration = (overrideArray.first?.duration ?? 0) as Decimal
  276. smbIsOff = overrideArray.first?.smbIsOff ?? false
  277. advancedSettings = overrideArray.first?.advancedSettings ?? false
  278. isfAndCr = overrideArray.first?.isfAndCr ?? true
  279. smbIsScheduledOff = overrideArray.first?.smbIsScheduledOff ?? false
  280. if advancedSettings {
  281. if !isfAndCr {
  282. isf = overrideArray.first?.isf ?? false
  283. cr = overrideArray.first?.cr ?? false
  284. }
  285. if smbIsScheduledOff {
  286. start = (overrideArray.first?.start ?? 0) as Decimal
  287. end = (overrideArray.first?.end ?? 0) as Decimal
  288. }
  289. if (overrideArray[0].smbMinutes as Decimal?) != nil {
  290. smbMinutes = (overrideArray.first?.smbMinutes ?? 30) as Decimal
  291. }
  292. if (overrideArray[0].uamMinutes as Decimal?) != nil {
  293. uamMinutes = (overrideArray.first?.uamMinutes ?? 30) as Decimal
  294. }
  295. }
  296. let overrideTarget = (overrideArray.first?.target ?? 0) as Decimal
  297. var newDuration = Double(duration)
  298. if isEnabled {
  299. let duration = overrideArray.first?.duration ?? 0
  300. let addedMinutes = Int(duration as Decimal)
  301. let date = overrideArray.first?.date ?? Date()
  302. if date.addingTimeInterval(addedMinutes.minutes.timeInterval) < Date(), !_indefinite {
  303. isEnabled = false
  304. }
  305. newDuration = Date().distance(to: date.addingTimeInterval(addedMinutes.minutes.timeInterval)).minutes
  306. if overrideTarget != 0 {
  307. override_target = true
  308. target = units == .mmolL ? overrideTarget.asMmolL : overrideTarget
  309. }
  310. }
  311. if newDuration < 0 { newDuration = 0 } else { duration = Decimal(newDuration) }
  312. if !isEnabled {
  313. _indefinite = true
  314. percentage = 100
  315. duration = 0
  316. target = 0
  317. override_target = false
  318. smbIsOff = false
  319. advancedSettings = false
  320. smbMinutes = defaultSmbMinutes
  321. uamMinutes = defaultUamMinutes
  322. }
  323. }
  324. }
  325. func populateSettings(from preset: OverridePresets) {
  326. profileName = preset.name ?? ""
  327. percentage = preset.percentage
  328. duration = (preset.duration ?? 0) as Decimal
  329. _indefinite = preset.indefinite
  330. override_target = preset.target != nil
  331. if let targetValue = preset.target as NSDecimalNumber? {
  332. target = units == .mmolL ? (targetValue as Decimal).asMmolL : targetValue as Decimal
  333. } else {
  334. target = 0
  335. }
  336. advancedSettings = preset.advancedSettings
  337. smbIsOff = preset.smbIsOff
  338. smbIsScheduledOff = preset.smbIsScheduledOff
  339. isf = preset.isf
  340. cr = preset.cr
  341. smbMinutes = (preset.smbMinutes ?? 0) as Decimal
  342. uamMinutes = (preset.uamMinutes ?? 0) as Decimal
  343. isfAndCr = preset.isfAndCr
  344. start = (preset.start ?? 0) as Decimal
  345. end = (preset.end ?? 0) as Decimal
  346. }
  347. func cancelProfile() {
  348. _indefinite = true
  349. isEnabled = false
  350. percentage = 100
  351. duration = 0
  352. target = 0
  353. override_target = false
  354. smbIsOff = false
  355. advancedSettings = false
  356. coredataContext.perform { [self] in
  357. let profiles = Override(context: self.coredataContext)
  358. profiles.enabled = false
  359. profiles.date = Date()
  360. try? self.coredataContext.save()
  361. >>>>>>> 9672da256c317a314acc76d6e4f6e82cc174d133
  362. }
  363. smbMinutes = defaultSmbMinutes
  364. uamMinutes = defaultUamMinutes
  365. }
  366. func updatePreset(_ preset: OverridePresets) {
  367. let context = CoreDataStack.shared.persistentContainer.viewContext
  368. context.performAndWait {
  369. preset.name = profileName
  370. preset.percentage = percentage
  371. preset.duration = NSDecimalNumber(decimal: duration)
  372. let targetValue = override_target ? (units == .mmolL ? target.asMgdL : target) : nil
  373. preset.target = targetValue != nil ? NSDecimalNumber(decimal: targetValue!) : nil
  374. preset.indefinite = _indefinite
  375. preset.advancedSettings = advancedSettings
  376. preset.smbIsOff = smbIsOff
  377. preset.smbIsScheduledOff = smbIsScheduledOff
  378. preset.isf = isf
  379. preset.cr = cr
  380. preset.smbMinutes = NSDecimalNumber(decimal: smbMinutes)
  381. preset.uamMinutes = NSDecimalNumber(decimal: uamMinutes)
  382. preset.isfAndCr = isfAndCr
  383. try? context.save()
  384. }
  385. }
  386. }
  387. }
  388. // MARK: - Setup Notifications
  389. extension OverrideProfilesConfig.StateModel {
  390. // Custom Notification to update View when an Override has been cancelled via Home View
  391. func setupNotification() {
  392. Foundation.NotificationCenter.default.addObserver(
  393. self,
  394. selector: #selector(handleOverrideConfigurationUpdate),
  395. name: .didUpdateOverrideConfiguration,
  396. object: nil
  397. )
  398. }
  399. @objc private func handleOverrideConfigurationUpdate() {
  400. updateLatestOverrideConfiguration()
  401. }
  402. // MARK: - Enact Overrides
  403. func reorderOverride(from source: IndexSet, to destination: Int) {
  404. overridePresets.move(fromOffsets: source, toOffset: destination)
  405. for (index, override) in overridePresets.enumerated() {
  406. override.orderPosition = Int16(index + 1)
  407. }
  408. do {
  409. guard viewContext.hasChanges else { return }
  410. try viewContext.save()
  411. // Update Presets View
  412. setupOverridePresetsArray()
  413. } catch {
  414. debugPrint(
  415. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to save after reordering Override Presets with error: \(error.localizedDescription)"
  416. )
  417. }
  418. }
  419. /// here we only have to update the Boolean Flag 'enabled'
  420. @MainActor func enactOverridePreset(withID id: NSManagedObjectID) async {
  421. do {
  422. /// get the underlying NSManagedObject of the Override that should be enabled
  423. let overrideToEnact = try viewContext.existingObject(with: id) as? OverrideStored
  424. overrideToEnact?.enabled = true
  425. overrideToEnact?.date = Date()
  426. overrideToEnact?.isUploadedToNS = false
  427. /// Update the 'Cancel Override' button state
  428. isEnabled = true
  429. /// disable all active Overrides and reset state variables
  430. /// do not create a OverrideRunEntry because we only want that if we cancel a running Override, not when enacting a Preset
  431. await disableAllActiveOverrides(except: id, createOverrideRunEntry: currentActiveOverride != nil)
  432. await resetStateVariables()
  433. guard viewContext.hasChanges else { return }
  434. try viewContext.save()
  435. // Update View
  436. updateLatestOverrideConfiguration()
  437. } catch {
  438. debugPrint("\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to enact Override Preset")
  439. }
  440. }
  441. // MARK: - Save the Override that we want to cancel to the OverrideRunStored Entity, then cancel ALL active overrides
  442. @MainActor func disableAllActiveOverrides(except overrideID: NSManagedObjectID? = nil, createOverrideRunEntry: Bool) async {
  443. // Get ALL NSManagedObject IDs of ALL active Override to cancel every single Override
  444. let ids = await overrideStorage.loadLatestOverrideConfigurations(fetchLimit: 0) // 0 = no fetch limit
  445. await viewContext.perform {
  446. do {
  447. // Fetch the existing OverrideStored objects from the context
  448. let results = try ids.compactMap { id in
  449. try self.viewContext.existingObject(with: id) as? OverrideStored
  450. }
  451. // If there are no results, return early
  452. guard !results.isEmpty else { return }
  453. // Check if we also need to create a corresponding OverrideRunStored entry, i.e. when the User uses the Cancel Button in Override View
  454. if createOverrideRunEntry {
  455. // Use the first override to create a new OverrideRunStored entry
  456. if let canceledOverride = results.first {
  457. let newOverrideRunStored = OverrideRunStored(context: self.viewContext)
  458. newOverrideRunStored.id = UUID()
  459. newOverrideRunStored.name = canceledOverride.name
  460. newOverrideRunStored.startDate = canceledOverride.date ?? .distantPast
  461. newOverrideRunStored.endDate = Date()
  462. newOverrideRunStored
  463. .target = NSDecimalNumber(decimal: self.overrideStorage.calculateTarget(override: canceledOverride))
  464. newOverrideRunStored.override = canceledOverride
  465. newOverrideRunStored.isUploadedToNS = false
  466. }
  467. }
  468. // Disable all override except the one with overrideID
  469. for overrideToCancel in results {
  470. if overrideToCancel.objectID != overrideID {
  471. overrideToCancel.enabled = false
  472. }
  473. }
  474. // Save the context if there are changes
  475. if self.viewContext.hasChanges {
  476. try self.viewContext.save()
  477. // Update the View
  478. self.updateLatestOverrideConfiguration()
  479. }
  480. } catch {
  481. debugPrint(
  482. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to disable active Overrides with error: \(error.localizedDescription)"
  483. )
  484. }
  485. }
  486. }
  487. // MARK: - Override (presets) save operations
  488. // Saves a Custom Override in a background context
  489. /// not a Preset
  490. func saveCustomOverride() async {
  491. let override = Override(
  492. name: overrideName,
  493. enabled: true,
  494. date: Date(),
  495. duration: overrideDuration,
  496. indefinite: indefinite,
  497. percentage: overrideSliderPercentage,
  498. smbIsOff: smbIsOff,
  499. isPreset: isPreset,
  500. id: id,
  501. overrideTarget: shouldOverrideTarget,
  502. target: target,
  503. advancedSettings: advancedSettings,
  504. isfAndCr: isfAndCr,
  505. isf: isf,
  506. cr: cr,
  507. smbIsAlwaysOff: smbIsAlwaysOff,
  508. start: start,
  509. end: end,
  510. smbMinutes: smbMinutes,
  511. uamMinutes: uamMinutes
  512. )
  513. // First disable all Overrides
  514. await disableAllActiveOverrides(createOverrideRunEntry: true)
  515. // Then save and activate a new custom Override and reset the State variables
  516. async let storeOverride: () = overrideStorage.storeOverride(override: override)
  517. async let resetState: () = resetStateVariables()
  518. _ = await (storeOverride, resetState)
  519. // Update View
  520. updateLatestOverrideConfiguration()
  521. }
  522. // Save Presets
  523. /// enabled has to be false, isPreset has to be true
  524. func saveOverridePreset() async {
  525. let preset = Override(
  526. name: overrideName,
  527. enabled: false,
  528. date: Date(),
  529. duration: overrideDuration,
  530. indefinite: indefinite,
  531. percentage: overrideSliderPercentage,
  532. smbIsOff: smbIsOff,
  533. isPreset: true,
  534. id: id,
  535. overrideTarget: shouldOverrideTarget,
  536. target: target,
  537. advancedSettings: advancedSettings,
  538. isfAndCr: isfAndCr,
  539. isf: isf,
  540. cr: cr,
  541. smbIsAlwaysOff: smbIsAlwaysOff,
  542. start: start,
  543. end: end,
  544. smbMinutes: smbMinutes,
  545. uamMinutes: uamMinutes
  546. )
  547. async let storeOverride: () = overrideStorage.storeOverride(override: preset)
  548. async let resetState: () = resetStateVariables()
  549. _ = await (storeOverride, resetState)
  550. // Update Presets View
  551. setupOverridePresetsArray()
  552. }
  553. // MARK: - Setup Override Presets Array
  554. // Fill the array of the Override Presets to display them in the UI
  555. private func setupOverridePresetsArray() {
  556. Task {
  557. let ids = await self.overrideStorage.fetchForOverridePresets()
  558. await updateOverridePresetsArray(with: ids)
  559. }
  560. }
  561. @MainActor private func updateOverridePresetsArray(with IDs: [NSManagedObjectID]) async {
  562. do {
  563. let overrideObjects = try IDs.compactMap { id in
  564. try viewContext.existingObject(with: id) as? OverrideStored
  565. }
  566. overridePresets = overrideObjects
  567. } catch {
  568. debugPrint(
  569. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to extract Overrides as NSManagedObjects from the NSManagedObjectIDs with error: \(error.localizedDescription)"
  570. )
  571. }
  572. }
  573. // MARK: - Override Preset Deletion
  574. func invokeOverridePresetDeletion(_ objectID: NSManagedObjectID) async {
  575. await overrideStorage.deleteOverridePreset(objectID)
  576. // Update Presets View
  577. setupOverridePresetsArray()
  578. }
  579. // MARK: - Setup the State variables with the last Override configuration
  580. /// First get the latest Overrides corresponding NSManagedObjectID with a background fetch
  581. /// Then unpack it on the view context and update the State variables which can be used on in the View for some Logic
  582. /// This also needs to be called when we cancel an Override via the Home View to update the State of the Button for this case
  583. func updateLatestOverrideConfiguration() {
  584. Task {
  585. let id = await overrideStorage.loadLatestOverrideConfigurations(fetchLimit: 1)
  586. async let updateState: () = updateLatestOverrideConfigurationOfState(from: id)
  587. async let setOverride: () = setCurrentOverride(from: id)
  588. _ = await (updateState, setOverride)
  589. }
  590. }
  591. @MainActor func updateLatestOverrideConfigurationOfState(from IDs: [NSManagedObjectID]) async {
  592. do {
  593. let result = try IDs.compactMap { id in
  594. try viewContext.existingObject(with: id) as? OverrideStored
  595. }
  596. isEnabled = result.first?.enabled ?? false
  597. if !isEnabled {
  598. await resetStateVariables()
  599. }
  600. } catch {
  601. debugPrint(
  602. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to updateLatestOverrideConfiguration"
  603. )
  604. }
  605. }
  606. // Sets the current active Preset name to show in the UI
  607. @MainActor func setCurrentOverride(from IDs: [NSManagedObjectID]) async {
  608. do {
  609. guard let firstID = IDs.first else {
  610. activeOverrideName = "Custom Override"
  611. currentActiveOverride = nil
  612. return
  613. }
  614. if let overrideToEdit = try viewContext.existingObject(with: firstID) as? OverrideStored {
  615. currentActiveOverride = overrideToEdit
  616. activeOverrideName = overrideToEdit.name ?? "Custom Override"
  617. }
  618. } catch {
  619. debugPrint(
  620. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to set active preset name with error: \(error.localizedDescription)"
  621. )
  622. }
  623. }
  624. @MainActor func duplicateOverridePresetAndCancelPreviousOverride() async {
  625. // We get the current active Preset by using currentActiveOverride which can either be a Preset or a custom Override
  626. guard let overridePresetToDuplicate = currentActiveOverride, overridePresetToDuplicate.isPreset == true else { return }
  627. // Copy the current Override-Preset to not edit the underlying Preset
  628. let duplidateId = await overrideStorage.copyRunningOverride(overridePresetToDuplicate)
  629. // Cancel the duplicated Override
  630. /// As we are on the Main Thread already we don't need to cancel via the objectID in this case
  631. do {
  632. try await viewContext.perform {
  633. overridePresetToDuplicate.enabled = false
  634. guard self.viewContext.hasChanges else { return }
  635. try self.viewContext.save()
  636. }
  637. // Update View
  638. // TODO: -
  639. if let overrideToEdit = try viewContext.existingObject(with: duplidateId) as? OverrideStored
  640. {
  641. currentActiveOverride = overrideToEdit
  642. activeOverrideName = overrideToEdit.name ?? "Custom Override"
  643. }
  644. } catch {
  645. debugPrint(
  646. "\(DebuggingIdentifiers.failed) \(#file) \(#function) Failed to cancel previous override with error: \(error.localizedDescription)"
  647. )
  648. }
  649. }
  650. // MARK: - Helper functions for Overrides
  651. @MainActor func resetStateVariables() async {
  652. id = ""
  653. overrideDuration = 0
  654. indefinite = true
  655. overrideSliderPercentage = 100
  656. advancedSettings = false
  657. smbIsOff = false
  658. overrideName = ""
  659. shouldOverrideTarget = false
  660. isf = true
  661. cr = true
  662. isfAndCr = true
  663. smbIsAlwaysOff = false
  664. start = 0
  665. end = 23
  666. smbMinutes = defaultSmbMinutes
  667. uamMinutes = defaultUamMinutes
  668. target = 0
  669. }
  670. }
  671. // MARK: - TEMP TARGET
  672. extension OverrideProfilesConfig.StateModel {
  673. func enact() {
  674. guard durationTT > 0 else {
  675. return
  676. }
  677. var lowTarget = low
  678. if viewPercantage {
  679. lowTarget = Decimal(round(Double(computeTarget())))
  680. coredataContext.performAndWait {
  681. let saveToCoreData = TempTargets(context: self.coredataContext)
  682. saveToCoreData.id = UUID().uuidString
  683. saveToCoreData.active = true
  684. saveToCoreData.hbt = hbt
  685. saveToCoreData.date = Date()
  686. saveToCoreData.duration = durationTT as NSDecimalNumber
  687. saveToCoreData.startDate = Date()
  688. try? self.coredataContext.save()
  689. }
  690. didSaveSettings = true
  691. } else {
  692. coredataContext.performAndWait {
  693. let saveToCoreData = TempTargets(context: coredataContext)
  694. saveToCoreData.active = false
  695. saveToCoreData.date = Date()
  696. do {
  697. guard coredataContext.hasChanges else { return }
  698. try coredataContext.save()
  699. } catch {
  700. print(error.localizedDescription)
  701. }
  702. }
  703. }
  704. var highTarget = lowTarget
  705. if units == .mmolL, !viewPercantage {
  706. lowTarget = Decimal(round(Double(lowTarget.asMgdL)))
  707. highTarget = lowTarget
  708. }
  709. let entry = TempTarget(
  710. name: TempTarget.custom,
  711. createdAt: date,
  712. targetTop: highTarget,
  713. targetBottom: lowTarget,
  714. duration: durationTT,
  715. enteredBy: TempTarget.manual,
  716. reason: TempTarget.custom
  717. )
  718. storage.storeTempTargets([entry])
  719. showModal(for: nil)
  720. }
  721. func cancel() {
  722. storage.storeTempTargets([TempTarget.cancel(at: Date())])
  723. showModal(for: nil)
  724. coredataContext.performAndWait {
  725. let saveToCoreData = TempTargets(context: self.coredataContext)
  726. saveToCoreData.active = false
  727. saveToCoreData.date = Date()
  728. do {
  729. guard coredataContext.hasChanges else { return }
  730. try coredataContext.save()
  731. } catch {
  732. print(error.localizedDescription)
  733. }
  734. let setHBT = TempTargetsSlider(context: self.coredataContext)
  735. setHBT.enabled = false
  736. setHBT.date = Date()
  737. do {
  738. guard coredataContext.hasChanges else { return }
  739. try coredataContext.save()
  740. } catch {
  741. print(error.localizedDescription)
  742. }
  743. }
  744. }
  745. func save() {
  746. guard durationTT > 0 else {
  747. return
  748. }
  749. var lowTarget = low
  750. if viewPercantage {
  751. lowTarget = Decimal(round(Double(computeTarget())))
  752. didSaveSettings = true
  753. }
  754. var highTarget = lowTarget
  755. if units == .mmolL, !viewPercantage {
  756. lowTarget = Decimal(round(Double(lowTarget.asMgdL)))
  757. highTarget = lowTarget
  758. }
  759. let entry = TempTarget(
  760. name: newPresetName.isEmpty ? TempTarget.custom : newPresetName,
  761. createdAt: Date(),
  762. targetTop: highTarget,
  763. targetBottom: lowTarget,
  764. duration: durationTT,
  765. enteredBy: TempTarget.manual,
  766. reason: newPresetName.isEmpty ? TempTarget.custom : newPresetName
  767. )
  768. presetsTT.append(entry)
  769. storage.storePresets(presetsTT)
  770. if viewPercantage {
  771. let id = entry.id
  772. coredataContext.performAndWait {
  773. let saveToCoreData = TempTargetsSlider(context: self.coredataContext)
  774. saveToCoreData.id = id
  775. saveToCoreData.isPreset = true
  776. saveToCoreData.enabled = true
  777. saveToCoreData.hbt = hbt
  778. saveToCoreData.date = Date()
  779. saveToCoreData.duration = durationTT as NSDecimalNumber
  780. do {
  781. guard coredataContext.hasChanges else { return }
  782. try coredataContext.save()
  783. } catch {
  784. print(error.localizedDescription)
  785. }
  786. }
  787. }
  788. }
  789. func enactPreset(id: String) {
  790. if var preset = presetsTT.first(where: { $0.id == id }) {
  791. preset.createdAt = Date()
  792. storage.storeTempTargets([preset])
  793. showModal(for: nil)
  794. coredataContext.performAndWait {
  795. var tempTargetsArray = [TempTargetsSlider]()
  796. let requestTempTargets = TempTargetsSlider.fetchRequest() as NSFetchRequest<TempTargetsSlider>
  797. let sortTT = NSSortDescriptor(key: "date", ascending: false)
  798. requestTempTargets.sortDescriptors = [sortTT]
  799. try? tempTargetsArray = coredataContext.fetch(requestTempTargets)
  800. let whichID = tempTargetsArray.first(where: { $0.id == id })
  801. if whichID != nil {
  802. let saveToCoreData = TempTargets(context: self.coredataContext)
  803. saveToCoreData.active = true
  804. saveToCoreData.date = Date()
  805. saveToCoreData.hbt = whichID?.hbt ?? 160
  806. // saveToCoreData.id = id
  807. saveToCoreData.startDate = Date()
  808. saveToCoreData.duration = whichID?.duration ?? 0
  809. do {
  810. guard coredataContext.hasChanges else { return }
  811. try coredataContext.save()
  812. } catch {
  813. print(error.localizedDescription)
  814. }
  815. } else {
  816. let saveToCoreData = TempTargets(context: self.coredataContext)
  817. saveToCoreData.active = false
  818. saveToCoreData.date = Date()
  819. do {
  820. guard coredataContext.hasChanges else { return }
  821. try coredataContext.save()
  822. } catch {
  823. print(error.localizedDescription)
  824. }
  825. }
  826. }
  827. }
  828. }
  829. func removePreset(id: String) {
  830. presetsTT = presetsTT.filter { $0.id != id }
  831. storage.storePresets(presetsTT)
  832. }
  833. func computeTarget() -> Decimal {
  834. var ratio = Decimal(percentageTT / 100)
  835. let c = Decimal(hbt - 100)
  836. var target = (c / ratio) - c + 100
  837. if c * (c + target - 100) <= 0 {
  838. ratio = maxValue
  839. target = (c / ratio) - c + 100
  840. }
  841. return Decimal(Double(target))
  842. }
  843. }
  844. extension OverrideProfilesConfig.StateModel: SettingsObserver {
  845. func settingsDidChange(_: FreeAPSSettings) {
  846. units = settingsManager.settings.units
  847. defaultSmbMinutes = settingsManager.preferences.maxSMBBasalMinutes
  848. defaultUamMinutes = settingsManager.preferences.maxUAMSMBBasalMinutes
  849. maxValue = settingsManager.preferences.autosensMax
  850. }
  851. }