- Add custom queue command mechanism to replace abusing getPumpStatus for this purpose
- Display error and confirmation dialog for commands executed from the Omnipod tab - Improve Exception logging in AapsOmnipodManager - A few random improvements and bug fixes
This commit is contained in:
parent
c7687aa6c7
commit
feecfc5676
37 changed files with 827 additions and 401 deletions
|
@ -26,4 +26,5 @@ abstract class CommandQueueModule {
|
||||||
@ContributesAndroidInjector abstract fun commandTempBasalAbsoluteInjector(): CommandTempBasalAbsolute
|
@ContributesAndroidInjector abstract fun commandTempBasalAbsoluteInjector(): CommandTempBasalAbsolute
|
||||||
@ContributesAndroidInjector abstract fun commandTempBasalPercentInjector(): CommandTempBasalPercent
|
@ContributesAndroidInjector abstract fun commandTempBasalPercentInjector(): CommandTempBasalPercent
|
||||||
@ContributesAndroidInjector abstract fun commandSetUserSettingsInjector(): CommandSetUserSettings
|
@ContributesAndroidInjector abstract fun commandSetUserSettingsInjector(): CommandSetUserSettings
|
||||||
|
@ContributesAndroidInjector abstract fun commandCustomCommandInjector(): CommandCustomCommand
|
||||||
}
|
}
|
|
@ -29,6 +29,7 @@ import info.nightscout.androidaps.db.CareportalEvent;
|
||||||
import info.nightscout.androidaps.db.Source;
|
import info.nightscout.androidaps.db.Source;
|
||||||
import info.nightscout.androidaps.db.TDD;
|
import info.nightscout.androidaps.db.TDD;
|
||||||
import info.nightscout.androidaps.db.TemporaryBasal;
|
import info.nightscout.androidaps.db.TemporaryBasal;
|
||||||
|
import info.nightscout.androidaps.db.Treatment;
|
||||||
import info.nightscout.androidaps.events.EventInitializationChanged;
|
import info.nightscout.androidaps.events.EventInitializationChanged;
|
||||||
import info.nightscout.androidaps.events.EventRefreshOverview;
|
import info.nightscout.androidaps.events.EventRefreshOverview;
|
||||||
import info.nightscout.androidaps.interfaces.CommandQueueProvider;
|
import info.nightscout.androidaps.interfaces.CommandQueueProvider;
|
||||||
|
@ -36,6 +37,7 @@ import info.nightscout.androidaps.interfaces.Constraint;
|
||||||
import info.nightscout.androidaps.interfaces.ConstraintsInterface;
|
import info.nightscout.androidaps.interfaces.ConstraintsInterface;
|
||||||
import info.nightscout.androidaps.interfaces.PluginDescription;
|
import info.nightscout.androidaps.interfaces.PluginDescription;
|
||||||
import info.nightscout.androidaps.interfaces.PluginType;
|
import info.nightscout.androidaps.interfaces.PluginType;
|
||||||
|
import info.nightscout.androidaps.interfaces.ProfileFunction;
|
||||||
import info.nightscout.androidaps.interfaces.PumpDescription;
|
import info.nightscout.androidaps.interfaces.PumpDescription;
|
||||||
import info.nightscout.androidaps.interfaces.PumpInterface;
|
import info.nightscout.androidaps.interfaces.PumpInterface;
|
||||||
import info.nightscout.androidaps.interfaces.PumpPluginBase;
|
import info.nightscout.androidaps.interfaces.PumpPluginBase;
|
||||||
|
@ -43,9 +45,9 @@ import info.nightscout.androidaps.logging.AAPSLogger;
|
||||||
import info.nightscout.androidaps.logging.LTag;
|
import info.nightscout.androidaps.logging.LTag;
|
||||||
import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.interfaces.ProfileFunction;
|
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventOverviewBolusProgress;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventOverviewBolusProgress;
|
||||||
|
@ -64,7 +66,6 @@ import info.nightscout.androidaps.plugins.pump.combo.ruffyscripter.history.PumpH
|
||||||
import info.nightscout.androidaps.plugins.pump.combo.ruffyscripter.history.PumpHistoryRequest;
|
import info.nightscout.androidaps.plugins.pump.combo.ruffyscripter.history.PumpHistoryRequest;
|
||||||
import info.nightscout.androidaps.plugins.pump.combo.ruffyscripter.history.Tdd;
|
import info.nightscout.androidaps.plugins.pump.combo.ruffyscripter.history.Tdd;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
||||||
import info.nightscout.androidaps.db.Treatment;
|
|
||||||
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin;
|
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin;
|
||||||
import info.nightscout.androidaps.utils.DateUtil;
|
import info.nightscout.androidaps.utils.DateUtil;
|
||||||
import info.nightscout.androidaps.utils.InstanceId;
|
import info.nightscout.androidaps.utils.InstanceId;
|
||||||
|
@ -1395,7 +1396,10 @@ public class ComboPlugin extends PumpPluginBase implements PumpInterface, Constr
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void executeCustomAction(CustomActionType customActionType) {
|
public void executeCustomAction(CustomActionType customActionType) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable @Override public PumpEnactResult executeCustomCommand(CustomCommand customCommand) {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -12,6 +12,7 @@ import android.os.IBinder;
|
||||||
import android.os.Looper;
|
import android.os.Looper;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
@ -56,6 +57,7 @@ import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.general.nsclient.NSUpload;
|
import info.nightscout.androidaps.plugins.general.nsclient.NSUpload;
|
||||||
import info.nightscout.androidaps.plugins.general.nsclient.UploadQueue;
|
import info.nightscout.androidaps.plugins.general.nsclient.UploadQueue;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
||||||
|
@ -1171,7 +1173,10 @@ public class LocalInsightPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void executeCustomAction(CustomActionType customActionType) {
|
public void executeCustomAction(CustomActionType customActionType) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable @Override public PumpEnactResult executeCustomCommand(CustomCommand customCommand) {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void readHistory() {
|
private void readHistory() {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package info.nightscout.androidaps.plugins.pump.mdi;
|
package info.nightscout.androidaps.plugins.pump.mdi;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
@ -27,6 +28,7 @@ import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
||||||
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin;
|
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin;
|
||||||
import info.nightscout.androidaps.utils.DateUtil;
|
import info.nightscout.androidaps.utils.DateUtil;
|
||||||
|
@ -281,7 +283,10 @@ public class MDIPlugin extends PumpPluginBase implements PumpInterface {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void executeCustomAction(CustomActionType customActionType) {
|
public void executeCustomAction(CustomActionType customActionType) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable @Override public PumpEnactResult executeCustomCommand(CustomCommand customCommand) {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -16,9 +16,9 @@ import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.bus.RxBusWrapper
|
import info.nightscout.androidaps.plugins.bus.RxBusWrapper
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType
|
import info.nightscout.androidaps.plugins.common.ManufacturerType
|
||||||
import info.nightscout.androidaps.interfaces.ProfileFunction
|
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventOverviewBolusProgress
|
import info.nightscout.androidaps.plugins.general.overview.events.EventOverviewBolusProgress
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
||||||
|
@ -133,6 +133,11 @@ class VirtualPumpPlugin @Inject constructor(
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun executeCustomAction(customActionType: CustomActionType) {}
|
override fun executeCustomAction(customActionType: CustomActionType) {}
|
||||||
|
|
||||||
|
override fun executeCustomCommand(customCommand: CustomCommand?): PumpEnactResult? {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
override fun isInitialized(): Boolean {
|
override fun isInitialized(): Boolean {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,16 +20,16 @@ import info.nightscout.androidaps.events.EventProfileNeedsUpdate
|
||||||
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
||||||
import info.nightscout.androidaps.interfaces.CommandQueueProvider
|
import info.nightscout.androidaps.interfaces.CommandQueueProvider
|
||||||
import info.nightscout.androidaps.interfaces.Constraint
|
import info.nightscout.androidaps.interfaces.Constraint
|
||||||
|
import info.nightscout.androidaps.interfaces.ProfileFunction
|
||||||
import info.nightscout.androidaps.logging.AAPSLogger
|
import info.nightscout.androidaps.logging.AAPSLogger
|
||||||
import info.nightscout.androidaps.logging.LTag
|
import info.nightscout.androidaps.logging.LTag
|
||||||
import info.nightscout.androidaps.plugins.bus.RxBusWrapper
|
import info.nightscout.androidaps.plugins.bus.RxBusWrapper
|
||||||
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
||||||
import info.nightscout.androidaps.interfaces.ProfileFunction
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissBolusProgressIfRunning
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissBolusProgressIfRunning
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
||||||
import info.nightscout.androidaps.queue.commands.Command
|
|
||||||
import info.nightscout.androidaps.queue.commands.*
|
import info.nightscout.androidaps.queue.commands.*
|
||||||
import info.nightscout.androidaps.queue.commands.Command.CommandType
|
import info.nightscout.androidaps.queue.commands.Command.CommandType
|
||||||
import info.nightscout.androidaps.utils.FabricPrivacy
|
import info.nightscout.androidaps.utils.FabricPrivacy
|
||||||
|
@ -485,6 +485,55 @@ class CommandQueue @Inject constructor(
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun customCommand(customCommand: CustomCommand, callback: Callback?): Boolean {
|
||||||
|
if (isCustomCommandInQueue(customCommand.javaClass)) {
|
||||||
|
callback?.result(executingNowError())?.run()
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// remove all unfinished
|
||||||
|
removeAllCustomCommands(customCommand.javaClass)
|
||||||
|
// add new command to queue
|
||||||
|
add(CommandCustomCommand(injector, customCommand, callback))
|
||||||
|
notifyAboutNewCommand()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
@Synchronized
|
||||||
|
override fun isCustomCommandInQueue(customCommandType: Class<out CustomCommand>): Boolean {
|
||||||
|
if(isCustomCommandRunning(customCommandType)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
synchronized(queue) {
|
||||||
|
for (i in queue.indices) {
|
||||||
|
val command = queue[i]
|
||||||
|
if (command is CommandCustomCommand && customCommandType.isInstance(command.customCommand)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun isCustomCommandRunning(customCommandType: Class<out CustomCommand>): Boolean {
|
||||||
|
val performing = this.performing
|
||||||
|
if (performing is CommandCustomCommand && customCommandType.isInstance(performing.customCommand)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
@Synchronized
|
||||||
|
private fun removeAllCustomCommands(targetType: Class<out CustomCommand>) {
|
||||||
|
synchronized(queue) {
|
||||||
|
for (i in queue.indices.reversed()) {
|
||||||
|
val command = queue[i]
|
||||||
|
if (command is CustomCommand && targetType.isInstance(command.commandType)) {
|
||||||
|
queue.removeAt(i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
override fun spannedStatus(): Spanned {
|
override fun spannedStatus(): Spanned {
|
||||||
var s = ""
|
var s = ""
|
||||||
var line = 0
|
var line = 0
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
package info.nightscout.androidaps.queue.commands
|
||||||
|
|
||||||
|
import dagger.android.HasAndroidInjector
|
||||||
|
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
||||||
|
import info.nightscout.androidaps.logging.LTag
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
|
import info.nightscout.androidaps.queue.Callback
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
class CommandCustomCommand(
|
||||||
|
injector: HasAndroidInjector,
|
||||||
|
val customCommand: CustomCommand,
|
||||||
|
callback: Callback?
|
||||||
|
) : Command(injector, CommandType.CUSTOM_COMMAND, callback) {
|
||||||
|
|
||||||
|
@Inject lateinit var activePlugin: ActivePluginProvider
|
||||||
|
|
||||||
|
override fun execute() {
|
||||||
|
val result = activePlugin.activePump.executeCustomCommand(customCommand)
|
||||||
|
aapsLogger.debug(LTag.PUMPQUEUE, "Result success: ${result?.success} enacted: ${result?.enacted}")
|
||||||
|
callback?.result(result)?.run()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun status(): String {
|
||||||
|
return customCommand.statusDescription
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,8 +2,6 @@ package info.nightscout.androidaps.queue.commands
|
||||||
|
|
||||||
import dagger.android.HasAndroidInjector
|
import dagger.android.HasAndroidInjector
|
||||||
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
||||||
import info.nightscout.androidaps.interfaces.ProfileFunction
|
|
||||||
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType
|
|
||||||
import info.nightscout.androidaps.plugins.pump.insight.LocalInsightPlugin
|
import info.nightscout.androidaps.plugins.pump.insight.LocalInsightPlugin
|
||||||
import info.nightscout.androidaps.queue.Callback
|
import info.nightscout.androidaps.queue.Callback
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
|
@ -14,17 +12,12 @@ class CommandStartPump(
|
||||||
) : Command(injector, CommandType.START_PUMP, callback) {
|
) : Command(injector, CommandType.START_PUMP, callback) {
|
||||||
|
|
||||||
@Inject lateinit var activePlugin: ActivePluginProvider
|
@Inject lateinit var activePlugin: ActivePluginProvider
|
||||||
@Inject lateinit var profileFunction: ProfileFunction
|
|
||||||
|
|
||||||
override fun execute() {
|
override fun execute() {
|
||||||
val pump = activePlugin.activePump
|
val pump = activePlugin.activePump
|
||||||
if (pump is LocalInsightPlugin) {
|
if (pump is LocalInsightPlugin) {
|
||||||
val result = pump.startPump()
|
val result = pump.startPump()
|
||||||
callback?.result(result)?.run()
|
callback?.result(result)?.run()
|
||||||
} else if (pump.pumpDescription.pumpType == PumpType.Insulet_Omnipod) {
|
|
||||||
// When using CommandQueue.setProfile, it refuses to set the profile is the same as the current profile
|
|
||||||
// However we need to set the current profile to resume delivery in case the Pod is suspended
|
|
||||||
pump.setNewBasalProfile(profileFunction.getProfile())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,7 @@ import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
||||||
import info.nightscout.androidaps.interfaces.Constraint
|
import info.nightscout.androidaps.interfaces.Constraint
|
||||||
import info.nightscout.androidaps.interfaces.PumpDescription
|
import info.nightscout.androidaps.interfaces.PumpDescription
|
||||||
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
import info.nightscout.androidaps.plugins.pump.virtual.VirtualPumpPlugin
|
import info.nightscout.androidaps.plugins.pump.virtual.VirtualPumpPlugin
|
||||||
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin
|
import info.nightscout.androidaps.plugins.treatments.TreatmentsPlugin
|
||||||
import info.nightscout.androidaps.queue.commands.Command
|
import info.nightscout.androidaps.queue.commands.Command
|
||||||
|
@ -192,4 +193,88 @@ class CommandQueueTest : TestBaseWithProfile() {
|
||||||
Assert.assertFalse(queued)
|
Assert.assertFalse(queued)
|
||||||
Assert.assertEquals(commandQueue.size(), 0)
|
Assert.assertEquals(commandQueue.size(), 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun isCustomCommandRunning() {
|
||||||
|
// given
|
||||||
|
Assert.assertEquals(0, commandQueue.size())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val queued1 = commandQueue.customCommand(CustomCommand1(), null)
|
||||||
|
val queued2 = commandQueue.customCommand(CustomCommand2(), null)
|
||||||
|
commandQueue.pickup()
|
||||||
|
|
||||||
|
// then
|
||||||
|
Assert.assertTrue(queued1)
|
||||||
|
Assert.assertTrue(queued2)
|
||||||
|
Assert.assertTrue(commandQueue.isCustomCommandInQueue(CustomCommand1::class.java))
|
||||||
|
Assert.assertTrue(commandQueue.isCustomCommandInQueue(CustomCommand2::class.java))
|
||||||
|
Assert.assertTrue(commandQueue.isCustomCommandRunning(CustomCommand1::class.java))
|
||||||
|
Assert.assertFalse(commandQueue.isCustomCommandRunning(CustomCommand2::class.java))
|
||||||
|
|
||||||
|
Assert.assertEquals(1, commandQueue.size())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun isCustomCommandInQueue() {
|
||||||
|
// given
|
||||||
|
Assert.assertEquals(0, commandQueue.size())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val queued1 = commandQueue.customCommand(CustomCommand1(), null)
|
||||||
|
val queued2 = commandQueue.customCommand(CustomCommand2(), null)
|
||||||
|
|
||||||
|
// then
|
||||||
|
Assert.assertTrue(queued1)
|
||||||
|
Assert.assertTrue(queued2)
|
||||||
|
Assert.assertTrue(commandQueue.isCustomCommandInQueue(CustomCommand1::class.java))
|
||||||
|
Assert.assertTrue(commandQueue.isCustomCommandInQueue(CustomCommand2::class.java))
|
||||||
|
Assert.assertFalse(commandQueue.isCustomCommandInQueue(CustomCommand3::class.java))
|
||||||
|
Assert.assertEquals(2, commandQueue.size())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun differentCustomCommandsAllowed() {
|
||||||
|
// given
|
||||||
|
Assert.assertEquals(0, commandQueue.size())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val queued1 = commandQueue.customCommand(CustomCommand1(), null)
|
||||||
|
val queued2 = commandQueue.customCommand(CustomCommand2(), null)
|
||||||
|
|
||||||
|
// then
|
||||||
|
Assert.assertTrue(queued1)
|
||||||
|
Assert.assertTrue(queued2)
|
||||||
|
Assert.assertEquals(2, commandQueue.size())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun sameCustomCommandNotAllowed() {
|
||||||
|
// given
|
||||||
|
Assert.assertEquals(0, commandQueue.size())
|
||||||
|
|
||||||
|
// when
|
||||||
|
val queued1 = commandQueue.customCommand(CustomCommand1(), null)
|
||||||
|
val queued2 = commandQueue.customCommand(CustomCommand1(), null)
|
||||||
|
|
||||||
|
// then
|
||||||
|
Assert.assertTrue(queued1)
|
||||||
|
Assert.assertFalse(queued2)
|
||||||
|
Assert.assertEquals(1, commandQueue.size())
|
||||||
|
}
|
||||||
|
|
||||||
|
private class CustomCommand1 : CustomCommand {
|
||||||
|
override val statusDescription: String
|
||||||
|
get() = "CUSTOM COMMAND 1"
|
||||||
|
}
|
||||||
|
|
||||||
|
private class CustomCommand2 : CustomCommand {
|
||||||
|
override val statusDescription: String
|
||||||
|
get() = "CUSTOM COMMAND 2"
|
||||||
|
}
|
||||||
|
|
||||||
|
private class CustomCommand3 : CustomCommand {
|
||||||
|
override val statusDescription: String
|
||||||
|
get() = "CUSTOM COMMAND 3"
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -3,6 +3,7 @@ package info.nightscout.androidaps.interfaces
|
||||||
import android.text.Spanned
|
import android.text.Spanned
|
||||||
import info.nightscout.androidaps.data.DetailedBolusInfo
|
import info.nightscout.androidaps.data.DetailedBolusInfo
|
||||||
import info.nightscout.androidaps.data.Profile
|
import info.nightscout.androidaps.data.Profile
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
import info.nightscout.androidaps.queue.Callback
|
import info.nightscout.androidaps.queue.Callback
|
||||||
import info.nightscout.androidaps.queue.commands.Command
|
import info.nightscout.androidaps.queue.commands.Command
|
||||||
|
|
||||||
|
@ -33,6 +34,9 @@ interface CommandQueueProvider {
|
||||||
fun setUserOptions(callback: Callback?): Boolean
|
fun setUserOptions(callback: Callback?): Boolean
|
||||||
fun loadTDDs(callback: Callback?): Boolean
|
fun loadTDDs(callback: Callback?): Boolean
|
||||||
fun loadEvents(callback: Callback?): Boolean
|
fun loadEvents(callback: Callback?): Boolean
|
||||||
|
fun customCommand(customCommand: CustomCommand, callback: Callback?): Boolean
|
||||||
|
fun isCustomCommandRunning(customCommandType: Class<out CustomCommand>): Boolean
|
||||||
|
fun isCustomCommandInQueue(customCommandType: Class<out CustomCommand>): Boolean
|
||||||
fun spannedStatus(): Spanned
|
fun spannedStatus(): Spanned
|
||||||
fun isThisProfileSet(profile: Profile): Boolean
|
fun isThisProfileSet(profile: Profile): Boolean
|
||||||
}
|
}
|
|
@ -1,5 +1,7 @@
|
||||||
package info.nightscout.androidaps.interfaces;
|
package info.nightscout.androidaps.interfaces;
|
||||||
|
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
|
||||||
|
@ -11,7 +13,9 @@ import info.nightscout.androidaps.data.PumpEnactResult;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
import info.nightscout.androidaps.plugins.pump.common.defs.PumpType;
|
||||||
|
import info.nightscout.androidaps.queue.Callback;
|
||||||
import info.nightscout.androidaps.utils.TimeChangeType;
|
import info.nightscout.androidaps.utils.TimeChangeType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -105,10 +109,32 @@ public interface PumpInterface {
|
||||||
|
|
||||||
boolean canHandleDST();
|
boolean canHandleDST();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides a list of custom actions to be displayed in the Actions tab.
|
||||||
|
* Plese note that these actions will not be queued upon execution
|
||||||
|
*
|
||||||
|
* @return list of custom actions
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
List<CustomAction> getCustomActions();
|
List<CustomAction> getCustomActions();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executes a custom action. Please not that these actions will not be queued
|
||||||
|
*
|
||||||
|
* @param customActionType action to be executed
|
||||||
|
*/
|
||||||
void executeCustomAction(CustomActionType customActionType);
|
void executeCustomAction(CustomActionType customActionType);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executes a custom queued command
|
||||||
|
* To place a custom command on the queue, see {@link CommandQueueProvider#customCommand(CustomCommand, Callback)}
|
||||||
|
*
|
||||||
|
* @param customCommand the custom command to be executed
|
||||||
|
* @return PumpEnactResult that represents the command execution result
|
||||||
|
*/
|
||||||
|
@Nullable
|
||||||
|
PumpEnactResult executeCustomCommand(CustomCommand customCommand);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method will be called when time or Timezone changes, and pump driver can then do a specific action (for
|
* This method will be called when time or Timezone changes, and pump driver can then do a specific action (for
|
||||||
* example update clock on pump).
|
* example update clock on pump).
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
package info.nightscout.androidaps.plugins.general.command.defs
|
||||||
|
|
||||||
|
import java.io.Serializable
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implement this interface for every custom pump command that you want to be able to queue
|
||||||
|
* See [info.nightscout.androidaps.interfaces.CommandQueueProvider] for queuing a custom command.
|
||||||
|
*/
|
||||||
|
interface CustomCommand : Serializable {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return short description of this command to be used in [info.nightscout.androidaps.queue.commands.Command.status]
|
||||||
|
* The description is typically all caps.
|
||||||
|
*/
|
||||||
|
val statusDescription: String
|
||||||
|
}
|
|
@ -31,7 +31,8 @@ abstract class Command(
|
||||||
SET_USER_SETTINGS, // so far only Dana specific,
|
SET_USER_SETTINGS, // so far only Dana specific,
|
||||||
START_PUMP,
|
START_PUMP,
|
||||||
STOP_PUMP,
|
STOP_PUMP,
|
||||||
INSIGHT_SET_TBR_OVER_ALARM // insight only
|
INSIGHT_SET_TBR_OVER_ALARM, // insight only
|
||||||
|
CUSTOM_COMMAND
|
||||||
}
|
}
|
||||||
|
|
||||||
init {
|
init {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package info.nightscout.androidaps.danar;
|
package info.nightscout.androidaps.danar;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
@ -9,9 +10,9 @@ import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import dagger.android.HasAndroidInjector;
|
import dagger.android.HasAndroidInjector;
|
||||||
import info.nightscout.androidaps.dana.DanaPumpInterface;
|
|
||||||
import info.nightscout.androidaps.dana.DanaFragment;
|
import info.nightscout.androidaps.dana.DanaFragment;
|
||||||
import info.nightscout.androidaps.dana.DanaPump;
|
import info.nightscout.androidaps.dana.DanaPump;
|
||||||
|
import info.nightscout.androidaps.dana.DanaPumpInterface;
|
||||||
import info.nightscout.androidaps.dana.comm.RecordTypes;
|
import info.nightscout.androidaps.dana.comm.RecordTypes;
|
||||||
import info.nightscout.androidaps.danar.services.AbstractDanaRExecutionService;
|
import info.nightscout.androidaps.danar.services.AbstractDanaRExecutionService;
|
||||||
import info.nightscout.androidaps.data.Profile;
|
import info.nightscout.androidaps.data.Profile;
|
||||||
|
@ -37,6 +38,7 @@ import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker;
|
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
||||||
|
@ -511,7 +513,10 @@ public abstract class AbstractDanaRPlugin extends PumpPluginBase implements Pump
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void executeCustomAction(CustomActionType customActionType) {
|
public void executeCustomAction(CustomActionType customActionType) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable @Override public PumpEnactResult executeCustomCommand(CustomCommand customCommand) {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -26,6 +26,7 @@ import info.nightscout.androidaps.plugins.common.ManufacturerType
|
||||||
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification
|
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification
|
||||||
|
@ -658,6 +659,7 @@ class DanaRSPlugin @Inject constructor(
|
||||||
override fun loadTDDs(): PumpEnactResult = loadHistory(info.nightscout.androidaps.dana.comm.RecordTypes.RECORD_TYPE_DAILY)
|
override fun loadTDDs(): PumpEnactResult = loadHistory(info.nightscout.androidaps.dana.comm.RecordTypes.RECORD_TYPE_DAILY)
|
||||||
override fun getCustomActions(): List<CustomAction>? = null
|
override fun getCustomActions(): List<CustomAction>? = null
|
||||||
override fun executeCustomAction(customActionType: CustomActionType) {}
|
override fun executeCustomAction(customActionType: CustomActionType) {}
|
||||||
|
override fun executeCustomCommand(customCommand: CustomCommand?): PumpEnactResult? = null
|
||||||
override fun canHandleDST(): Boolean = false
|
override fun canHandleDST(): Boolean = false
|
||||||
override fun timezoneOrDSTChanged(timeChangeType: TimeChangeType?) {}
|
override fun timezoneOrDSTChanged(timeChangeType: TimeChangeType?) {}
|
||||||
override fun clearPairing() {
|
override fun clearPairing() {
|
||||||
|
|
|
@ -48,6 +48,7 @@ import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.PumpPluginAbstract;
|
import info.nightscout.androidaps.plugins.pump.common.PumpPluginAbstract;
|
||||||
|
@ -1583,6 +1584,10 @@ public class MedtronicPumpPlugin extends PumpPluginAbstract implements PumpInter
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Nullable @Override public PumpEnactResult executeCustomCommand(CustomCommand customCommand) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void timezoneOrDSTChanged(TimeChangeType changeType) {
|
public void timezoneOrDSTChanged(TimeChangeType changeType) {
|
||||||
|
|
||||||
|
|
|
@ -12,21 +12,16 @@ import android.os.SystemClock;
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.joda.time.DateTime;
|
import org.joda.time.DateTime;
|
||||||
import org.joda.time.Duration;
|
import org.joda.time.Duration;
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
import org.json.JSONObject;
|
import org.json.JSONObject;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.EnumSet;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.Set;
|
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
|
@ -47,6 +42,7 @@ import info.nightscout.androidaps.interfaces.ActivePluginProvider;
|
||||||
import info.nightscout.androidaps.interfaces.CommandQueueProvider;
|
import info.nightscout.androidaps.interfaces.CommandQueueProvider;
|
||||||
import info.nightscout.androidaps.interfaces.PluginDescription;
|
import info.nightscout.androidaps.interfaces.PluginDescription;
|
||||||
import info.nightscout.androidaps.interfaces.PluginType;
|
import info.nightscout.androidaps.interfaces.PluginType;
|
||||||
|
import info.nightscout.androidaps.interfaces.ProfileFunction;
|
||||||
import info.nightscout.androidaps.interfaces.PumpDescription;
|
import info.nightscout.androidaps.interfaces.PumpDescription;
|
||||||
import info.nightscout.androidaps.interfaces.PumpInterface;
|
import info.nightscout.androidaps.interfaces.PumpInterface;
|
||||||
import info.nightscout.androidaps.interfaces.PumpPluginBase;
|
import info.nightscout.androidaps.interfaces.PumpPluginBase;
|
||||||
|
@ -56,7 +52,7 @@ import info.nightscout.androidaps.plugins.bus.RxBusWrapper;
|
||||||
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
import info.nightscout.androidaps.plugins.common.ManufacturerType;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomAction;
|
||||||
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
import info.nightscout.androidaps.plugins.general.actions.defs.CustomActionType;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventDismissNotification;
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
import info.nightscout.androidaps.plugins.general.overview.events.EventNewNotification;
|
||||||
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
import info.nightscout.androidaps.plugins.general.overview.notifications.Notification;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.data.TempBasalPair;
|
import info.nightscout.androidaps.plugins.pump.common.data.TempBasalPair;
|
||||||
|
@ -74,7 +70,6 @@ import info.nightscout.androidaps.plugins.pump.omnipod.data.ActiveBolus;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.data.RLHistoryItemOmnipod;
|
import info.nightscout.androidaps.plugins.pump.omnipod.data.RLHistoryItemOmnipod;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodCommandType;
|
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodCommandType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodCustomActionType;
|
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodCustomActionType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodStatusRequestType;
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodStorageKeys;
|
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodStorageKeys;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.action.service.ExpirationReminderBuilder;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.action.service.ExpirationReminderBuilder;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.message.response.podinfo.PodInfoRecentPulseLog;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.message.response.podinfo.PodInfoRecentPulseLog;
|
||||||
|
@ -84,6 +79,10 @@ import info.nightscout.androidaps.plugins.pump.omnipod.driver.manager.PodStateMa
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodPumpValuesChanged;
|
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodPumpValuesChanged;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodTbrChanged;
|
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodTbrChanged;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.manager.AapsOmnipodManager;
|
import info.nightscout.androidaps.plugins.pump.omnipod.manager.AapsOmnipodManager;
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.queue.command.CommandHandleTimeChange;
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.queue.command.CommandUpdateAlertConfiguration;
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.queue.command.OmnipodCustomCommand;
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.queue.command.OmnipodCustomCommandType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.rileylink.service.RileyLinkOmnipodService;
|
import info.nightscout.androidaps.plugins.pump.omnipod.rileylink.service.RileyLinkOmnipodService;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.ui.OmnipodFragment;
|
import info.nightscout.androidaps.plugins.pump.omnipod.ui.OmnipodFragment;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.util.AapsOmnipodUtil;
|
import info.nightscout.androidaps.plugins.pump.omnipod.util.AapsOmnipodUtil;
|
||||||
|
@ -107,6 +106,7 @@ import io.reactivex.schedulers.Schedulers;
|
||||||
public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface, RileyLinkPumpDevice {
|
public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface, RileyLinkPumpDevice {
|
||||||
private static final long RILEY_LINK_CONNECT_TIMEOUT_MILLIS = 3 * 60 * 1000L; // 3 minutes
|
private static final long RILEY_LINK_CONNECT_TIMEOUT_MILLIS = 3 * 60 * 1000L; // 3 minutes
|
||||||
private static final long STATUS_CHECK_INTERVAL_MILLIS = 60 * 1000L; // 1 minute
|
private static final long STATUS_CHECK_INTERVAL_MILLIS = 60 * 1000L; // 1 minute
|
||||||
|
public static final int STARTUP_STATUS_REQUEST_TRIES = 3;
|
||||||
|
|
||||||
private final PodStateManager podStateManager;
|
private final PodStateManager podStateManager;
|
||||||
private final RileyLinkServiceData rileyLinkServiceData;
|
private final RileyLinkServiceData rileyLinkServiceData;
|
||||||
|
@ -115,6 +115,7 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
private final AapsOmnipodUtil aapsOmnipodUtil;
|
private final AapsOmnipodUtil aapsOmnipodUtil;
|
||||||
private final RileyLinkUtil rileyLinkUtil;
|
private final RileyLinkUtil rileyLinkUtil;
|
||||||
private final OmnipodAlertUtil omnipodAlertUtil;
|
private final OmnipodAlertUtil omnipodAlertUtil;
|
||||||
|
private final ProfileFunction profileFunction;
|
||||||
private final AAPSLogger aapsLogger;
|
private final AAPSLogger aapsLogger;
|
||||||
private final RxBusWrapper rxBus;
|
private final RxBusWrapper rxBus;
|
||||||
private final ActivePluginProvider activePlugin;
|
private final ActivePluginProvider activePlugin;
|
||||||
|
@ -127,8 +128,8 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
private final ServiceConnection serviceConnection;
|
private final ServiceConnection serviceConnection;
|
||||||
private final PumpType pumpType = PumpType.Insulet_Omnipod;
|
private final PumpType pumpType = PumpType.Insulet_Omnipod;
|
||||||
|
|
||||||
private final List<CustomAction> customActions = new ArrayList<>();
|
private final List<CustomAction> customActions = Collections.singletonList(new CustomAction(
|
||||||
private final Set<OmnipodStatusRequestType> statusRequests = Collections.synchronizedSet(EnumSet.noneOf(OmnipodStatusRequestType.class));
|
R.string.omnipod_custom_action_reset_rileylink, OmnipodCustomActionType.RESET_RILEY_LINK_CONFIGURATION, true));
|
||||||
private final CompositeDisposable disposables = new CompositeDisposable();
|
private final CompositeDisposable disposables = new CompositeDisposable();
|
||||||
|
|
||||||
// variables for handling statuses and history
|
// variables for handling statuses and history
|
||||||
|
@ -139,12 +140,11 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
private boolean busy = false;
|
private boolean busy = false;
|
||||||
private int timeChangeRetries;
|
private int timeChangeRetries;
|
||||||
private long nextPodCheck;
|
private long nextPodCheck;
|
||||||
private boolean sentIdToFirebase;
|
|
||||||
private long lastConnectionTimeMillis;
|
private long lastConnectionTimeMillis;
|
||||||
private final Handler loopHandler = new Handler(Looper.getMainLooper());
|
private final Handler loopHandler = new Handler(Looper.getMainLooper());
|
||||||
|
|
||||||
private final Runnable statusChecker;
|
private final Runnable statusChecker;
|
||||||
private OmnipodCommandType currentCommand;
|
private boolean isCancelTempBasalRunning;
|
||||||
|
|
||||||
@Inject
|
@Inject
|
||||||
public OmnipodPumpPlugin(
|
public OmnipodPumpPlugin(
|
||||||
|
@ -164,7 +164,8 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
DateUtil dateUtil,
|
DateUtil dateUtil,
|
||||||
AapsOmnipodUtil aapsOmnipodUtil,
|
AapsOmnipodUtil aapsOmnipodUtil,
|
||||||
RileyLinkUtil rileyLinkUtil,
|
RileyLinkUtil rileyLinkUtil,
|
||||||
OmnipodAlertUtil omnipodAlertUtil
|
OmnipodAlertUtil omnipodAlertUtil,
|
||||||
|
ProfileFunction profileFunction
|
||||||
) {
|
) {
|
||||||
super(new PluginDescription() //
|
super(new PluginDescription() //
|
||||||
.mainType(PluginType.PUMP) //
|
.mainType(PluginType.PUMP) //
|
||||||
|
@ -189,12 +190,10 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
this.aapsOmnipodUtil = aapsOmnipodUtil;
|
this.aapsOmnipodUtil = aapsOmnipodUtil;
|
||||||
this.rileyLinkUtil = rileyLinkUtil;
|
this.rileyLinkUtil = rileyLinkUtil;
|
||||||
this.omnipodAlertUtil = omnipodAlertUtil;
|
this.omnipodAlertUtil = omnipodAlertUtil;
|
||||||
|
this.profileFunction = profileFunction;
|
||||||
|
|
||||||
pumpDescription = new PumpDescription(pumpType);
|
pumpDescription = new PumpDescription(pumpType);
|
||||||
|
|
||||||
customActions.add(new CustomAction(
|
|
||||||
R.string.omnipod_custom_action_reset_rileylink, OmnipodCustomActionType.RESET_RILEY_LINK_CONFIGURATION, true));
|
|
||||||
|
|
||||||
this.serviceConnection = new ServiceConnection() {
|
this.serviceConnection = new ServiceConnection() {
|
||||||
@Override
|
@Override
|
||||||
public void onServiceConnected(ComponentName name, IBinder service) {
|
public void onServiceConnected(ComponentName name, IBinder service) {
|
||||||
|
@ -231,14 +230,11 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
aapsOmnipodManager.createSuspendedFakeTbrIfNotExists();
|
aapsOmnipodManager.createSuspendedFakeTbrIfNotExists();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!getCommandQueue().statusInQueue()) {
|
if (OmnipodPumpPlugin.this.hasTimeDateOrTimeZoneChanged) {
|
||||||
if (!OmnipodPumpPlugin.this.statusRequests.isEmpty()) {
|
getCommandQueue().customCommand(new CommandHandleTimeChange(false), null);
|
||||||
getCommandQueue().readStatus("Status Refresh Requested", null);
|
|
||||||
} else if (OmnipodPumpPlugin.this.hasTimeDateOrTimeZoneChanged) {
|
|
||||||
getCommandQueue().readStatus("Date or Time Zone Changed", null);
|
|
||||||
} else if (!OmnipodPumpPlugin.this.verifyPodAlertConfiguration()) {
|
|
||||||
getCommandQueue().readStatus("Expiration Alerts Changed", null);
|
|
||||||
}
|
}
|
||||||
|
if (!OmnipodPumpPlugin.this.verifyPodAlertConfiguration()) {
|
||||||
|
getCommandQueue().customCommand(new CommandUpdateAlertConfiguration(), null);
|
||||||
}
|
}
|
||||||
|
|
||||||
doPodCheck();
|
doPodCheck();
|
||||||
|
@ -291,7 +287,7 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
event.isChanged(getResourceHelper(), R.string.key_omnipod_low_reservoir_alert_enabled) ||
|
event.isChanged(getResourceHelper(), R.string.key_omnipod_low_reservoir_alert_enabled) ||
|
||||||
event.isChanged(getResourceHelper(), R.string.key_omnipod_low_reservoir_alert_units)) {
|
event.isChanged(getResourceHelper(), R.string.key_omnipod_low_reservoir_alert_units)) {
|
||||||
if (!verifyPodAlertConfiguration() && !getCommandQueue().statusInQueue()) {
|
if (!verifyPodAlertConfiguration() && !getCommandQueue().statusInQueue()) {
|
||||||
getCommandQueue().readStatus("Expiration Alerts Changed", null);
|
getCommandQueue().customCommand(new CommandUpdateAlertConfiguration(), null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, fabricPrivacy::logException)
|
}, fabricPrivacy::logException)
|
||||||
|
@ -335,7 +331,7 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
// The pod is not running a TBR, while AAPS thinks it is
|
// The pod is not running a TBR, while AAPS thinks it is
|
||||||
if (!podStateManager.isTempBasalRunning()) {
|
if (!podStateManager.isTempBasalRunning()) {
|
||||||
// Only report TBR cancellations if they haven't been explicitly requested
|
// Only report TBR cancellations if they haven't been explicitly requested
|
||||||
if (currentCommand != OmnipodCommandType.CANCEL_TEMPORARY_BASAL) {
|
if (!isCancelTempBasalRunning) {
|
||||||
if (activePlugin.getActiveTreatments().isTempBasalInProgress() && !aapsOmnipodManager.hasSuspendedFakeTbr()) {
|
if (activePlugin.getActiveTreatments().isTempBasalInProgress() && !aapsOmnipodManager.hasSuspendedFakeTbr()) {
|
||||||
aapsOmnipodManager.reportCancelledTbr();
|
aapsOmnipodManager.reportCancelledTbr();
|
||||||
}
|
}
|
||||||
|
@ -361,13 +357,9 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
Notification notification = new Notification(Notification.OMNIPOD_POD_NOT_ATTACHED, resourceHelper.gs(R.string.omnipod_error_pod_not_attached), Notification.NORMAL);
|
Notification notification = new Notification(Notification.OMNIPOD_POD_NOT_ATTACHED, resourceHelper.gs(R.string.omnipod_error_pod_not_attached), Notification.NORMAL);
|
||||||
rxBus.send(new EventNewNotification(notification));
|
rxBus.send(new EventNewNotification(notification));
|
||||||
} else {
|
} else {
|
||||||
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_NOT_ATTACHED));
|
|
||||||
|
|
||||||
if (podStateManager.isSuspended()) {
|
if (podStateManager.isSuspended()) {
|
||||||
Notification notification = new Notification(Notification.OMNIPOD_POD_SUSPENDED, resourceHelper.gs(R.string.omnipod_error_pod_suspended), Notification.NORMAL);
|
Notification notification = new Notification(Notification.OMNIPOD_POD_SUSPENDED, resourceHelper.gs(R.string.omnipod_error_pod_suspended), Notification.NORMAL);
|
||||||
rxBus.send(new EventNewNotification(notification));
|
rxBus.send(new EventNewNotification(notification));
|
||||||
} else {
|
|
||||||
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_SUSPENDED));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -375,7 +367,6 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO is this correct?
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isInitialized() {
|
public boolean isInitialized() {
|
||||||
return isConnected() && podStateManager.isPodActivationCompleted();
|
return isConnected() && podStateManager.isPodActivationCompleted();
|
||||||
|
@ -455,130 +446,23 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
lastConnectionTimeMillis = System.currentTimeMillis();
|
lastConnectionTimeMillis = System.currentTimeMillis();
|
||||||
}
|
}
|
||||||
|
|
||||||
// We abuse getPumpStatus to squeeze commands in the queue
|
/**
|
||||||
// The only actual status requests we send to the Pod are on startup (in initializeAfterRileyLinkConnection)
|
* The only actual status requests we send here to the Pod are on startup (in initializeAfterRileyLinkConnection)
|
||||||
// And when the user explicitly requested it by clicking the Refresh button on the Omnipod tab
|
* And when the user explicitly requested it by clicking the Refresh button on the Omnipod tab (which is executed through {@link #executeCustomCommand(CustomCommand)})
|
||||||
// We don't do periodical status requests because that can drain the Pod's battery
|
* We don't do periodical status requests because that could drain the Pod's battery
|
||||||
// However that should be fine because we get a StatusResponse from all insulin commands sent to the Pod
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void getPumpStatus() {
|
public void getPumpStatus() {
|
||||||
if (firstRun) {
|
if (firstRun) {
|
||||||
initializeAfterRileyLinkConnection();
|
initializeAfterRileyLinkConnection();
|
||||||
} else if (!statusRequests.isEmpty()) {
|
firstRun = false;
|
||||||
synchronized (statusRequests) {
|
|
||||||
Iterator<OmnipodStatusRequestType> iterator = statusRequests.iterator();
|
|
||||||
|
|
||||||
while (iterator.hasNext()) {
|
|
||||||
OmnipodStatusRequestType statusRequest = iterator.next();
|
|
||||||
switch (statusRequest) {
|
|
||||||
case GET_PULSE_LOG:
|
|
||||||
try {
|
|
||||||
PodInfoRecentPulseLog result = executeCommand(OmnipodCommandType.GET_POD_PULSE_LOG, aapsOmnipodManager::readPulseLog);
|
|
||||||
Intent i = new Intent(context, ErrorHelperActivity.class);
|
|
||||||
i.putExtra("soundid", 0);
|
|
||||||
i.putExtra("status", "Pulse Log (copied to clipboard):\n" + result.toString());
|
|
||||||
i.putExtra("title", resourceHelper.gs(R.string.omnipod_warning));
|
|
||||||
i.putExtra("clipboardContent", result.toString());
|
|
||||||
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
||||||
context.startActivity(i);
|
|
||||||
} catch (Exception ex) {
|
|
||||||
aapsLogger.warn(LTag.PUMP, "Failed to retrieve pulse log", ex);
|
|
||||||
Intent i = new Intent(context, ErrorHelperActivity.class);
|
|
||||||
i.putExtra("soundid", 0);
|
|
||||||
i.putExtra("status", "Failed to retrieve pulse log");
|
|
||||||
i.putExtra("title", resourceHelper.gs(R.string.omnipod_warning));
|
|
||||||
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
||||||
context.startActivity(i);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case ACKNOWLEDGE_ALERTS:
|
|
||||||
executeCommand(OmnipodCommandType.ACKNOWLEDGE_ALERTS, aapsOmnipodManager::acknowledgeAlerts);
|
|
||||||
break;
|
|
||||||
case GET_POD_STATE:
|
|
||||||
executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
|
||||||
break;
|
|
||||||
case SUSPEND_DELIVERY:
|
|
||||||
executeCommand(OmnipodCommandType.SUSPEND_DELIVERY, aapsOmnipodManager::suspendDelivery);
|
|
||||||
break;
|
|
||||||
case SET_TIME:
|
|
||||||
executeCommand(OmnipodCommandType.SET_TIME, aapsOmnipodManager::setTime);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
aapsLogger.error(LTag.PUMP, "Unknown status request: " + statusRequest.name());
|
|
||||||
}
|
|
||||||
iterator.remove();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (this.hasTimeDateOrTimeZoneChanged) {
|
|
||||||
PumpEnactResult result;
|
|
||||||
if (aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
|
||||||
result = executeCommand(OmnipodCommandType.SET_TIME, aapsOmnipodManager::setTime);
|
|
||||||
} else {
|
|
||||||
// Even if automatically changing the time is disabled, we still want to at least do a GetStatus request,
|
|
||||||
// in order to update the Pod's activation time, which we need for calculating the time on the Pod
|
|
||||||
result = executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
this.hasTimeDateOrTimeZoneChanged = false;
|
|
||||||
timeChangeRetries = 0;
|
|
||||||
|
|
||||||
if (aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
|
||||||
Notification notification = new Notification(
|
|
||||||
Notification.TIME_OR_TIMEZONE_CHANGE,
|
|
||||||
resourceHelper.gs(R.string.omnipod_time_or_timezone_change),
|
|
||||||
Notification.INFO, 60);
|
|
||||||
rxBus.send(new EventNewNotification(notification));
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
timeChangeRetries++;
|
|
||||||
|
|
||||||
if (timeChangeRetries > 3) {
|
|
||||||
if (aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
|
||||||
Notification notification = new Notification(
|
|
||||||
Notification.TIME_OR_TIMEZONE_CHANGE,
|
|
||||||
resourceHelper.gs(R.string.omnipod_time_or_timezone_change_failed),
|
|
||||||
Notification.INFO, 60);
|
|
||||||
rxBus.send(new EventNewNotification(notification));
|
|
||||||
}
|
|
||||||
this.hasTimeDateOrTimeZoneChanged = false;
|
|
||||||
timeChangeRetries = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (!verifyPodAlertConfiguration()) {
|
|
||||||
Duration expirationReminderTimeBeforeShutdown = omnipodAlertUtil.getExpirationReminderTimeBeforeShutdown();
|
|
||||||
Integer lowReservoirAlertUnits = omnipodAlertUtil.getLowReservoirAlertUnits();
|
|
||||||
|
|
||||||
List<AlertConfiguration> alertConfigurations = new ExpirationReminderBuilder(podStateManager) //
|
|
||||||
.expirationAdvisory(expirationReminderTimeBeforeShutdown != null,
|
|
||||||
Optional.ofNullable(expirationReminderTimeBeforeShutdown).orElse(Duration.ZERO)) //
|
|
||||||
.lowReservoir(lowReservoirAlertUnits != null, Optional.ofNullable(lowReservoirAlertUnits).orElse(0)) //
|
|
||||||
.build();
|
|
||||||
|
|
||||||
PumpEnactResult result = executeCommand(OmnipodCommandType.CONFIGURE_ALERTS, () -> aapsOmnipodManager.configureAlerts(alertConfigurations));
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
aapsLogger.info(LTag.PUMP, "Successfully configured alerts in Pod");
|
|
||||||
|
|
||||||
podStateManager.setExpirationAlertTimeBeforeShutdown(expirationReminderTimeBeforeShutdown);
|
|
||||||
podStateManager.setLowReservoirAlertUnits(lowReservoirAlertUnits);
|
|
||||||
|
|
||||||
Notification notification = new Notification(
|
|
||||||
Notification.OMNIPOD_POD_ALERTS_UPDATED,
|
|
||||||
resourceHelper.gs(R.string.omnipod_expiration_alerts_updated),
|
|
||||||
Notification.INFO, 60);
|
|
||||||
rxBus.send(new EventNewNotification(notification));
|
|
||||||
} else {
|
|
||||||
aapsLogger.warn(LTag.PUMP, "Failed to configure alerts in Pod");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull
|
@NonNull
|
||||||
@Override
|
@Override
|
||||||
public PumpEnactResult setNewBasalProfile(Profile profile) {
|
public PumpEnactResult setNewBasalProfile(Profile profile) {
|
||||||
PumpEnactResult result = executeCommand(OmnipodCommandType.SET_BASAL_PROFILE, () -> aapsOmnipodManager.setBasalProfile(profile));
|
PumpEnactResult result = executeCommand(OmnipodCommandType.SET_BASAL_PROFILE, () -> aapsOmnipodManager.setBasalProfile(profile, true));
|
||||||
|
|
||||||
aapsLogger.info(LTag.PUMP, "Basal Profile was set: " + result.success);
|
aapsLogger.info(LTag.PUMP, "Basal Profile was set: " + result.success);
|
||||||
|
|
||||||
|
@ -597,7 +481,7 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long lastDataTime() {
|
public long lastDataTime() {
|
||||||
return podStateManager.isPodActivationCompleted() ? podStateManager.getLastSuccessfulCommunication().getMillis() : 0;
|
return podStateManager.isPodInitialized() ? podStateManager.getLastSuccessfulCommunication().getMillis() : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -652,7 +536,7 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
executeCommand(OmnipodCommandType.CANCEL_BOLUS, aapsOmnipodManager::cancelBolus);
|
executeCommand(OmnipodCommandType.CANCEL_BOLUS, aapsOmnipodManager::cancelBolus);
|
||||||
}
|
}
|
||||||
|
|
||||||
// if enforceNew===true current temp basal is cancelled and new TBR set (duration is prolonged),
|
// if enforceNew is true, current temp basal is cancelled and new TBR set (duration is prolonged),
|
||||||
// if false and the same rate is requested enacted=false and success=true is returned and TBR is not changed
|
// if false and the same rate is requested enacted=false and success=true is returned and TBR is not changed
|
||||||
@Override
|
@Override
|
||||||
public PumpEnactResult setTempBasalAbsolute(Double absoluteRate, Integer
|
public PumpEnactResult setTempBasalAbsolute(Double absoluteRate, Integer
|
||||||
|
@ -694,7 +578,12 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
return new PumpEnactResult(getInjector()).success(true).enacted(false);
|
return new PumpEnactResult(getInjector()).success(true).enacted(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isCancelTempBasalRunning = true;
|
||||||
|
try {
|
||||||
return executeCommand(OmnipodCommandType.CANCEL_TEMPORARY_BASAL, aapsOmnipodManager::cancelTemporaryBasal);
|
return executeCommand(OmnipodCommandType.CANCEL_TEMPORARY_BASAL, aapsOmnipodManager::cancelTemporaryBasal);
|
||||||
|
} finally {
|
||||||
|
isCancelTempBasalRunning = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO improve (i8n and more)
|
// TODO improve (i8n and more)
|
||||||
|
@ -753,12 +642,12 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
return pumpType.getManufacturer();
|
return pumpType.getManufacturer();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override @NotNull
|
@Override @NonNull
|
||||||
public PumpType model() {
|
public PumpType model() {
|
||||||
return pumpType;
|
return pumpType;
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull
|
@NonNull
|
||||||
@Override
|
@Override
|
||||||
public String serialNumber() {
|
public String serialNumber() {
|
||||||
return podStateManager.isPodInitialized() ? String.valueOf(podStateManager.getAddress()) : "-";
|
return podStateManager.isPodInitialized() ? String.valueOf(podStateManager.getAddress()) : "-";
|
||||||
|
@ -813,18 +702,143 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
aapsLogger.warn(LTag.PUMP, "Unknown custom action: {}" + mcat);
|
aapsLogger.warn(LTag.PUMP, "Unknown custom action: " + mcat);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PumpEnactResult executeCustomCommand(CustomCommand command) {
|
||||||
|
if (!(command instanceof OmnipodCustomCommand)) {
|
||||||
|
aapsLogger.warn(LTag.PUMP, "Unknown custom command: " + command.getClass().getName());
|
||||||
|
return new PumpEnactResult(getInjector()).success(false).enacted(false).comment(resourceHelper.gs(R.string.omnipod_unknown_custom_command, command.getClass().getName()));
|
||||||
|
}
|
||||||
|
|
||||||
|
OmnipodCustomCommandType commandType = ((OmnipodCustomCommand) command).getType();
|
||||||
|
|
||||||
|
aapsLogger.debug(LTag.PUMP, "Executing custom command: " + commandType);
|
||||||
|
|
||||||
|
switch (commandType) {
|
||||||
|
case ACKNOWLEDGE_ALERTS:
|
||||||
|
return executeCommand(OmnipodCommandType.ACKNOWLEDGE_ALERTS, aapsOmnipodManager::acknowledgeAlerts);
|
||||||
|
case GET_POD_STATUS:
|
||||||
|
return executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
||||||
|
case READ_PULSE_LOG:
|
||||||
|
return retrievePulseLog();
|
||||||
|
case SUSPEND_DELIVERY:
|
||||||
|
return executeCommand(OmnipodCommandType.SUSPEND_DELIVERY, aapsOmnipodManager::suspendDelivery);
|
||||||
|
case RESUME_DELIVERY:
|
||||||
|
return executeCommand(OmnipodCommandType.RESUME_DELIVERY, () -> aapsOmnipodManager.setBasalProfile(profileFunction.getProfile(), false));
|
||||||
|
case HANDLE_TIME_CHANGE:
|
||||||
|
return handleTimeChange(((CommandHandleTimeChange) command).isRequestedByUser());
|
||||||
|
case UPDATE_ALERT_CONFIGURATION:
|
||||||
|
return updateAlertConfiguration();
|
||||||
|
default:
|
||||||
|
aapsLogger.warn(LTag.PUMP, "Unknown custom command: " + commandType);
|
||||||
|
return new PumpEnactResult(getInjector()).success(false).enacted(false).comment(resourceHelper.gs(R.string.omnipod_unknown_custom_command, commandType));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private PumpEnactResult retrievePulseLog() {
|
||||||
|
PodInfoRecentPulseLog result;
|
||||||
|
try {
|
||||||
|
result = executeCommand(OmnipodCommandType.READ_POD_PULSE_LOG, aapsOmnipodManager::readPulseLog);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
return new PumpEnactResult(getInjector()).success(false).enacted(false).comment(aapsOmnipodManager.translateException(ex));
|
||||||
|
}
|
||||||
|
|
||||||
|
Intent i = new Intent(context, ErrorHelperActivity.class);
|
||||||
|
i.putExtra("soundid", 0);
|
||||||
|
i.putExtra("status", resourceHelper.gs(R.string.omnipod_pulse_log_value) + ":\n" + result.toString());
|
||||||
|
i.putExtra("title", resourceHelper.gs(R.string.omnipod_pulse_log));
|
||||||
|
i.putExtra("clipboardContent", result.toString());
|
||||||
|
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
||||||
|
context.startActivity(i);
|
||||||
|
return new PumpEnactResult(getInjector()).success(true).enacted(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull private PumpEnactResult updateAlertConfiguration() {
|
||||||
|
Duration expirationReminderTimeBeforeShutdown = omnipodAlertUtil.getExpirationReminderTimeBeforeShutdown();
|
||||||
|
Integer lowReservoirAlertUnits = omnipodAlertUtil.getLowReservoirAlertUnits();
|
||||||
|
|
||||||
|
List<AlertConfiguration> alertConfigurations = new ExpirationReminderBuilder(podStateManager) //
|
||||||
|
.expirationAdvisory(expirationReminderTimeBeforeShutdown != null,
|
||||||
|
Optional.ofNullable(expirationReminderTimeBeforeShutdown).orElse(Duration.ZERO)) //
|
||||||
|
.lowReservoir(lowReservoirAlertUnits != null, Optional.ofNullable(lowReservoirAlertUnits).orElse(0)) //
|
||||||
|
.build();
|
||||||
|
|
||||||
|
PumpEnactResult result = executeCommand(OmnipodCommandType.CONFIGURE_ALERTS, () -> aapsOmnipodManager.configureAlerts(alertConfigurations));
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
aapsLogger.info(LTag.PUMP, "Successfully configured alerts in Pod");
|
||||||
|
|
||||||
|
podStateManager.setExpirationAlertTimeBeforeShutdown(expirationReminderTimeBeforeShutdown);
|
||||||
|
podStateManager.setLowReservoirAlertUnits(lowReservoirAlertUnits);
|
||||||
|
|
||||||
|
Notification notification = new Notification(
|
||||||
|
Notification.OMNIPOD_POD_ALERTS_UPDATED,
|
||||||
|
resourceHelper.gs(R.string.omnipod_expiration_alerts_updated),
|
||||||
|
Notification.INFO, 60);
|
||||||
|
rxBus.send(new EventNewNotification(notification));
|
||||||
|
} else {
|
||||||
|
aapsLogger.warn(LTag.PUMP, "Failed to configure alerts in Pod");
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull private PumpEnactResult handleTimeChange(boolean requestedByUser) {
|
||||||
|
aapsLogger.debug(LTag.PUMP, "Setting time, requestedByUser={}", requestedByUser);
|
||||||
|
|
||||||
|
PumpEnactResult result;
|
||||||
|
if (requestedByUser || aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
||||||
|
result = executeCommand(OmnipodCommandType.SET_TIME, () -> aapsOmnipodManager.setTime(!requestedByUser));
|
||||||
|
} else {
|
||||||
|
// Even if automatically changing the time is disabled, we still want to at least do a GetStatus request,
|
||||||
|
// in order to update the Pod's activation time, which we need for calculating the time on the Pod
|
||||||
|
result = executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
this.hasTimeDateOrTimeZoneChanged = false;
|
||||||
|
timeChangeRetries = 0;
|
||||||
|
|
||||||
|
if (!requestedByUser && aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
||||||
|
Notification notification = new Notification(
|
||||||
|
Notification.TIME_OR_TIMEZONE_CHANGE,
|
||||||
|
resourceHelper.gs(R.string.omnipod_time_or_timezone_change),
|
||||||
|
Notification.INFO, 60);
|
||||||
|
rxBus.send(new EventNewNotification(notification));
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if (!requestedByUser) {
|
||||||
|
timeChangeRetries++;
|
||||||
|
|
||||||
|
if (timeChangeRetries > 3) {
|
||||||
|
if (aapsOmnipodManager.isTimeChangeEventEnabled()) {
|
||||||
|
Notification notification = new Notification(
|
||||||
|
Notification.TIME_OR_TIMEZONE_CHANGE,
|
||||||
|
resourceHelper.gs(R.string.omnipod_time_or_timezone_change_failed),
|
||||||
|
Notification.INFO, 60);
|
||||||
|
rxBus.send(new EventNewNotification(notification));
|
||||||
|
}
|
||||||
|
this.hasTimeDateOrTimeZoneChanged = false;
|
||||||
|
timeChangeRetries = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void timezoneOrDSTChanged(TimeChangeType timeChangeType) {
|
public void timezoneOrDSTChanged(TimeChangeType timeChangeType) {
|
||||||
aapsLogger.warn(LTag.PUMP, "Time, Date and/or TimeZone changed. [changeType=" + timeChangeType.name() + ", eventHandlingEnabled=" + aapsOmnipodManager.isTimeChangeEventEnabled() + "]");
|
aapsLogger.warn(LTag.PUMP, "Time, Date and/or TimeZone changed. [changeType=" + timeChangeType.name() + ", eventHandlingEnabled=" + aapsOmnipodManager.isTimeChangeEventEnabled() + "]");
|
||||||
|
|
||||||
if (podStateManager.isPodRunning()) {
|
if (podStateManager.isPodRunning()) {
|
||||||
aapsLogger.info(LTag.PUMP, "Time, Date and/or TimeZone changed event received and will be consumed by driver.");
|
aapsLogger.info(LTag.PUMP, "Time, Date and/or TimeZone changed event received and will be consumed by driver.");
|
||||||
this.hasTimeDateOrTimeZoneChanged = true;
|
hasTimeDateOrTimeZoneChanged = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -853,10 +867,6 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void addPodStatusRequest(OmnipodStatusRequestType pumpStatusRequest) {
|
|
||||||
statusRequests.add(pumpStatusRequest);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isFakingTempsByExtendedBoluses() {
|
public boolean isFakingTempsByExtendedBoluses() {
|
||||||
return false;
|
return false;
|
||||||
|
@ -887,54 +897,47 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
aapsLogger.debug(LTag.PUMP, "stopConnecting [PumpPluginAbstract] - default (empty) implementation.");
|
aapsLogger.debug(LTag.PUMP, "stopConnecting [PumpPluginAbstract] - default (empty) implementation.");
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull @Override public PumpEnactResult setTempBasalPercent(Integer percent, Integer
|
@NonNull @Override public PumpEnactResult setTempBasalPercent(Integer percent, Integer
|
||||||
durationInMinutes, Profile profile, boolean enforceNew) {
|
durationInMinutes, Profile profile, boolean enforceNew) {
|
||||||
aapsLogger.debug(LTag.PUMP, "setTempBasalPercent [OmnipodPumpPlugin] - Not implemented.");
|
aapsLogger.debug(LTag.PUMP, "setTempBasalPercent [OmnipodPumpPlugin] - Not implemented.");
|
||||||
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull @Override public PumpEnactResult setExtendedBolus(Double insulin, Integer
|
@NonNull @Override public PumpEnactResult setExtendedBolus(Double insulin, Integer
|
||||||
durationInMinutes) {
|
durationInMinutes) {
|
||||||
aapsLogger.debug(LTag.PUMP, "setExtendedBolus [OmnipodPumpPlugin] - Not implemented.");
|
aapsLogger.debug(LTag.PUMP, "setExtendedBolus [OmnipodPumpPlugin] - Not implemented.");
|
||||||
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull @Override public PumpEnactResult cancelExtendedBolus() {
|
@NonNull @Override public PumpEnactResult cancelExtendedBolus() {
|
||||||
aapsLogger.debug(LTag.PUMP, "cancelExtendedBolus [OmnipodPumpPlugin] - Not implemented.");
|
aapsLogger.debug(LTag.PUMP, "cancelExtendedBolus [OmnipodPumpPlugin] - Not implemented.");
|
||||||
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
@NotNull @Override public PumpEnactResult loadTDDs() {
|
@NonNull @Override public PumpEnactResult loadTDDs() {
|
||||||
aapsLogger.debug(LTag.PUMP, "loadTDDs [OmnipodPumpPlugin] - Not implemented.");
|
aapsLogger.debug(LTag.PUMP, "loadTDDs [OmnipodPumpPlugin] - Not implemented.");
|
||||||
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
return getOperationNotSupportedWithCustomText(info.nightscout.androidaps.core.R.string.pump_operation_not_supported_by_pump_driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
public OmnipodCommandType getCurrentCommand() {
|
|
||||||
return currentCommand;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initializeAfterRileyLinkConnection() {
|
private void initializeAfterRileyLinkConnection() {
|
||||||
if (podStateManager.isPodInitialized() && podStateManager.getPodProgressStatus().isAtLeast(PodProgressStatus.PAIRING_COMPLETED)) {
|
if (podStateManager.isPodInitialized() && podStateManager.getPodProgressStatus().isAtLeast(PodProgressStatus.PAIRING_COMPLETED)) {
|
||||||
PumpEnactResult result = executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
PumpEnactResult result = executeCommand(OmnipodCommandType.GET_POD_STATUS, aapsOmnipodManager::getPodStatus);
|
||||||
|
for (int i = 0; STARTUP_STATUS_REQUEST_TRIES > i; i++) {
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
aapsLogger.debug(LTag.PUMP, "Successfully retrieved Pod status on startup");
|
aapsLogger.debug(LTag.PUMP, "Successfully retrieved Pod status on startup");
|
||||||
|
break;
|
||||||
} else {
|
} else {
|
||||||
aapsLogger.warn(LTag.PUMP, "Failed to retrieve Pod status on startup");
|
aapsLogger.warn(LTag.PUMP, "Failed to retrieve Pod status on startup");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
aapsLogger.debug(LTag.PUMP, "Not retrieving Pod status on startup: no Pod running");
|
aapsLogger.debug(LTag.PUMP, "Not retrieving Pod status on startup: no Pod running");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!sentIdToFirebase) {
|
|
||||||
Bundle params = new Bundle();
|
Bundle params = new Bundle();
|
||||||
params.putString("version", BuildConfig.VERSION);
|
params.putString("version", BuildConfig.VERSION);
|
||||||
|
|
||||||
fabricPrivacy.getFirebaseAnalytics().logEvent("OmnipodPumpInit", params);
|
fabricPrivacy.getFirebaseAnalytics().logEvent("OmnipodPumpInit", params);
|
||||||
|
|
||||||
sentIdToFirebase = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.firstRun = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@NonNull private PumpEnactResult deliverBolus(final DetailedBolusInfo detailedBolusInfo) {
|
@NonNull private PumpEnactResult deliverBolus(final DetailedBolusInfo detailedBolusInfo) {
|
||||||
|
@ -952,14 +955,12 @@ public class OmnipodPumpPlugin extends PumpPluginBase implements PumpInterface,
|
||||||
|
|
||||||
private <T> T executeCommand(OmnipodCommandType commandType, Supplier<T> supplier) {
|
private <T> T executeCommand(OmnipodCommandType commandType, Supplier<T> supplier) {
|
||||||
try {
|
try {
|
||||||
currentCommand = commandType;
|
|
||||||
aapsLogger.debug(LTag.PUMP, "Executing command: {}", commandType);
|
aapsLogger.debug(LTag.PUMP, "Executing command: {}", commandType);
|
||||||
|
|
||||||
rileyLinkUtil.getRileyLinkHistory().add(new RLHistoryItemOmnipod(getInjector(), commandType));
|
rileyLinkUtil.getRileyLinkHistory().add(new RLHistoryItemOmnipod(getInjector(), commandType));
|
||||||
|
|
||||||
return supplier.get();
|
return supplier.get();
|
||||||
} finally {
|
} finally {
|
||||||
currentCommand = null;
|
|
||||||
rxBus.send(new EventRefreshOverview("Omnipod command: " + commandType.name(), false));
|
rxBus.send(new EventRefreshOverview("Omnipod command: " + commandType.name(), false));
|
||||||
rxBus.send(new EventOmnipodPumpValuesChanged());
|
rxBus.send(new EventOmnipodPumpValuesChanged());
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,8 +19,9 @@ public enum OmnipodCommandType {
|
||||||
SET_TIME(R.string.omnipod_cmd_set_time), //
|
SET_TIME(R.string.omnipod_cmd_set_time), //
|
||||||
CONFIGURE_ALERTS(R.string.omnipod_cmd_configure_alerts), //
|
CONFIGURE_ALERTS(R.string.omnipod_cmd_configure_alerts), //
|
||||||
ACKNOWLEDGE_ALERTS(R.string.omnipod_cmd_acknowledge_alerts), //
|
ACKNOWLEDGE_ALERTS(R.string.omnipod_cmd_acknowledge_alerts), //
|
||||||
GET_POD_PULSE_LOG(R.string.omnipod_cmd_get_pulse_log), //
|
READ_POD_PULSE_LOG(R.string.omnipod_cmd_get_pulse_log), //
|
||||||
SUSPEND_DELIVERY(R.string.omnipod_cmd_suspend_delivery);
|
SUSPEND_DELIVERY(R.string.omnipod_cmd_suspend_delivery),
|
||||||
|
RESUME_DELIVERY(R.string.omnipod_cmd_resume_delivery);
|
||||||
|
|
||||||
private int resourceId;
|
private int resourceId;
|
||||||
|
|
||||||
|
|
|
@ -1,9 +0,0 @@
|
||||||
package info.nightscout.androidaps.plugins.pump.omnipod.definition;
|
|
||||||
|
|
||||||
public enum OmnipodStatusRequestType {
|
|
||||||
ACKNOWLEDGE_ALERTS,
|
|
||||||
GET_POD_STATE,
|
|
||||||
GET_PULSE_LOG,
|
|
||||||
SUSPEND_DELIVERY,
|
|
||||||
SET_TIME
|
|
||||||
}
|
|
|
@ -17,11 +17,9 @@ import info.nightscout.androidaps.plugins.pump.omnipod.driver.util.AlertConfigur
|
||||||
public final class ExpirationReminderBuilder {
|
public final class ExpirationReminderBuilder {
|
||||||
private final Map<AlertSlot, AlertConfiguration> alerts = new HashMap<>();
|
private final Map<AlertSlot, AlertConfiguration> alerts = new HashMap<>();
|
||||||
private final DateTime endOfServiceTime;
|
private final DateTime endOfServiceTime;
|
||||||
private final PodStateManager podStateManager;
|
|
||||||
|
|
||||||
public ExpirationReminderBuilder(PodStateManager podStateManager) {
|
public ExpirationReminderBuilder(PodStateManager podStateManager) {
|
||||||
this.endOfServiceTime = podStateManager.getActivatedAt().plus(OmnipodConstants.SERVICE_DURATION);
|
this.endOfServiceTime = podStateManager.getActivatedAt().plus(OmnipodConstants.SERVICE_DURATION);
|
||||||
this.podStateManager = podStateManager;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public ExpirationReminderBuilder defaults() {
|
public ExpirationReminderBuilder defaults() {
|
||||||
|
@ -45,21 +43,17 @@ public final class ExpirationReminderBuilder {
|
||||||
public ExpirationReminderBuilder expirationAdvisory(boolean active, Duration timeBeforeShutdown) {
|
public ExpirationReminderBuilder expirationAdvisory(boolean active, Duration timeBeforeShutdown) {
|
||||||
DateTime expirationAdvisoryAlarmTime = endOfServiceTime.minus(timeBeforeShutdown);
|
DateTime expirationAdvisoryAlarmTime = endOfServiceTime.minus(timeBeforeShutdown);
|
||||||
|
|
||||||
if (DateTime.now().isBefore(expirationAdvisoryAlarmTime)) {
|
Duration timeUntilExpirationAdvisoryAlarm = DateTime.now().isBefore(expirationAdvisoryAlarmTime) ? new Duration(DateTime.now(),
|
||||||
Duration timeUntilExpirationAdvisoryAlarm = new Duration(DateTime.now(),
|
expirationAdvisoryAlarmTime) : Duration.ZERO;
|
||||||
expirationAdvisoryAlarmTime);
|
|
||||||
AlertConfiguration expirationAdvisoryAlertConfiguration = AlertConfigurationUtil.createExpirationAdvisoryAlertConfiguration(active,
|
AlertConfiguration expirationAdvisoryAlertConfiguration = AlertConfigurationUtil.createExpirationAdvisoryAlertConfiguration(active,
|
||||||
timeUntilExpirationAdvisoryAlarm, timeBeforeShutdown);
|
timeUntilExpirationAdvisoryAlarm, timeBeforeShutdown);
|
||||||
alerts.put(expirationAdvisoryAlertConfiguration.getAlertSlot(), expirationAdvisoryAlertConfiguration);
|
alerts.put(expirationAdvisoryAlertConfiguration.getAlertSlot(), expirationAdvisoryAlertConfiguration);
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public ExpirationReminderBuilder lowReservoir(boolean active, int units) {
|
public ExpirationReminderBuilder lowReservoir(boolean active, int units) {
|
||||||
if (podStateManager.getReservoirLevel() == null || podStateManager.getReservoirLevel().intValue() >= units) {
|
|
||||||
AlertConfiguration lowReservoirAlertConfiguration = AlertConfigurationUtil.createLowReservoirAlertConfiguration(active, (double) units);
|
AlertConfiguration lowReservoirAlertConfiguration = AlertConfigurationUtil.createLowReservoirAlertConfiguration(active, (double) units);
|
||||||
alerts.put(lowReservoirAlertConfiguration.getAlertSlot(), lowReservoirAlertConfiguration);
|
alerts.put(lowReservoirAlertConfiguration.getAlertSlot(), lowReservoirAlertConfiguration);
|
||||||
}
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,10 +3,13 @@ package info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.mes
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import info.nightscout.androidaps.logging.AAPSLogger;
|
||||||
|
import info.nightscout.androidaps.logging.LTag;
|
||||||
import info.nightscout.androidaps.plugins.pump.common.utils.ByteUtil;
|
import info.nightscout.androidaps.plugins.pump.common.utils.ByteUtil;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.message.command.GetStatusCommand;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.communication.message.command.GetStatusCommand;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.MessageBlockType;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.MessageBlockType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.OmnipodCrc;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.OmnipodCrc;
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.PacketType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.PodInfoType;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.PodInfoType;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.exception.CrcMismatchException;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.exception.CrcMismatchException;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.exception.MessageDecodingException;
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.exception.MessageDecodingException;
|
||||||
|
@ -91,8 +94,11 @@ public class OmnipodMessage {
|
||||||
return encodedData;
|
return encodedData;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void padWithGetStatusCommands(int packetSize) {
|
public void padWithGetStatusCommands(int packetSize, AAPSLogger aapsLogger) {
|
||||||
while (getEncoded().length < packetSize) {
|
while (getEncoded().length <= packetSize) {
|
||||||
|
if (getEncoded().length == PacketType.PDM.getMaxBodyLength()) {
|
||||||
|
aapsLogger.debug(LTag.PUMPBTCOMM, "Message length equals max body length: {}", this);
|
||||||
|
}
|
||||||
messageBlocks.add(new GetStatusCommand(PodInfoType.NORMAL));
|
messageBlocks.add(new GetStatusCommand(PodInfoType.NORMAL));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -224,10 +224,6 @@ public class OmnipodManager {
|
||||||
suspendDelivery(acknowledgementBeep);
|
suspendDelivery(acknowledgementBeep);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store the new Basal schedule after successfully suspending delivery, so that if setting the Basal schedule fails,
|
|
||||||
// And we later try to resume delivery, the new schedule is used
|
|
||||||
podStateManager.setBasalSchedule(schedule);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
executeAndVerify(() -> communicationService.executeAction(new SetBasalScheduleAction(podStateManager, schedule,
|
executeAndVerify(() -> communicationService.executeAction(new SetBasalScheduleAction(podStateManager, schedule,
|
||||||
false, podStateManager.getScheduleOffset(), acknowledgementBeep)));
|
false, podStateManager.getScheduleOffset(), acknowledgementBeep)));
|
||||||
|
@ -248,6 +244,8 @@ public class OmnipodManager {
|
||||||
throw ex;
|
throw ex;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
logCommandExecutionFinished("setBasalSchedule");
|
logCommandExecutionFinished("setBasalSchedule");
|
||||||
}
|
}
|
||||||
|
@ -420,7 +418,7 @@ public class OmnipodManager {
|
||||||
bolusCommandExecutionSubject = null;
|
bolusCommandExecutionSubject = null;
|
||||||
|
|
||||||
disposables.add(Completable.complete() //
|
disposables.add(Completable.complete() //
|
||||||
.delay(estimatedRemainingBolusDuration.getMillis() + 250, TimeUnit.MILLISECONDS) //
|
.delay(estimatedRemainingBolusDuration.getMillis(), TimeUnit.MILLISECONDS) //
|
||||||
.observeOn(Schedulers.io()) //
|
.observeOn(Schedulers.io()) //
|
||||||
.doOnComplete(() -> {
|
.doOnComplete(() -> {
|
||||||
synchronized (bolusDataMutex) {
|
synchronized (bolusDataMutex) {
|
||||||
|
@ -432,9 +430,8 @@ public class OmnipodManager {
|
||||||
StatusResponse statusResponse = getPodStatus();
|
StatusResponse statusResponse = getPodStatus();
|
||||||
if (statusResponse.getDeliveryStatus().isBolusing()) {
|
if (statusResponse.getDeliveryStatus().isBolusing()) {
|
||||||
throw new IllegalDeliveryStatusException(DeliveryStatus.NORMAL, statusResponse.getDeliveryStatus());
|
throw new IllegalDeliveryStatusException(DeliveryStatus.NORMAL, statusResponse.getDeliveryStatus());
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
} catch (PodFaultException ex) {
|
} catch (PodFaultException ex) {
|
||||||
// Subtract units not delivered in case of a Pod failure
|
// Subtract units not delivered in case of a Pod failure
|
||||||
bolusNotDelivered = ex.getFaultEvent().getBolusNotDelivered();
|
bolusNotDelivered = ex.getFaultEvent().getBolusNotDelivered();
|
||||||
|
|
|
@ -18,12 +18,12 @@ public class AlertConfigurationUtil {
|
||||||
|
|
||||||
public static AlertConfiguration createExpirationAdvisoryAlertConfiguration(boolean active, Duration timeUntilAlert, Duration duration) {
|
public static AlertConfiguration createExpirationAdvisoryAlertConfiguration(boolean active, Duration timeUntilAlert, Duration duration) {
|
||||||
return new AlertConfiguration(AlertType.EXPIRATION_ADVISORY_ALERT, AlertSlot.SLOT7, active, false, duration,
|
return new AlertConfiguration(AlertType.EXPIRATION_ADVISORY_ALERT, AlertSlot.SLOT7, active, false, duration,
|
||||||
new TimerAlertTrigger(timeUntilAlert), BeepType.BIP_BEEP_BIP_BEEP_BIP_BEEP_BIP_BEEP, BeepRepeat.EVERY_15_MINUTES);
|
new TimerAlertTrigger(timeUntilAlert), BeepType.BIP_BEEP_BIP_BEEP_BIP_BEEP_BIP_BEEP, BeepRepeat.EVERY_MINUTE_FOR_3_MINUTES_REPEAT_EVERY_15_MINUTES);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static AlertConfiguration createShutdownImminentAlertConfiguration(Duration timeUntilAlert) {
|
public static AlertConfiguration createShutdownImminentAlertConfiguration(Duration timeUntilAlert) {
|
||||||
return new AlertConfiguration(AlertType.SHUTDOWN_IMMINENT_ALARM, AlertSlot.SLOT2, true, false, Duration.ZERO,
|
return new AlertConfiguration(AlertType.SHUTDOWN_IMMINENT_ALARM, AlertSlot.SLOT2, true, false, Duration.ZERO,
|
||||||
new TimerAlertTrigger(timeUntilAlert), BeepType.BIP_BEEP_BIP_BEEP_BIP_BEEP_BIP_BEEP, BeepRepeat.EVERY_MINUTE_FOR_3_MINUTES_REPEAT_EVERY_15_MINUTES);
|
new TimerAlertTrigger(timeUntilAlert), BeepType.BIP_BEEP_BIP_BEEP_BIP_BEEP_BIP_BEEP, BeepRepeat.EVERY_15_MINUTES);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static AlertConfiguration createAutoOffAlertConfiguration(boolean active, Duration countdownDuration) {
|
public static AlertConfiguration createAutoOffAlertConfiguration(boolean active, Duration countdownDuration) {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import org.joda.time.Duration;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
import javax.inject.Singleton;
|
import javax.inject.Singleton;
|
||||||
|
@ -75,7 +76,6 @@ import info.nightscout.androidaps.plugins.pump.omnipod.util.AapsOmnipodUtil;
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.util.OmnipodAlertUtil;
|
import info.nightscout.androidaps.plugins.pump.omnipod.util.OmnipodAlertUtil;
|
||||||
import info.nightscout.androidaps.utils.resources.ResourceHelper;
|
import info.nightscout.androidaps.utils.resources.ResourceHelper;
|
||||||
import info.nightscout.androidaps.utils.sharedPreferences.SP;
|
import info.nightscout.androidaps.utils.sharedPreferences.SP;
|
||||||
import io.reactivex.disposables.Disposable;
|
|
||||||
import io.reactivex.subjects.SingleSubject;
|
import io.reactivex.subjects.SingleSubject;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
|
@ -120,11 +120,11 @@ public class AapsOmnipodManager {
|
||||||
this.aapsOmnipodUtil = aapsOmnipodUtil;
|
this.aapsOmnipodUtil = aapsOmnipodUtil;
|
||||||
this.aapsLogger = aapsLogger;
|
this.aapsLogger = aapsLogger;
|
||||||
this.rxBus = rxBus;
|
this.rxBus = rxBus;
|
||||||
|
this.sp = sp;
|
||||||
this.resourceHelper = resourceHelper;
|
this.resourceHelper = resourceHelper;
|
||||||
this.injector = injector;
|
this.injector = injector;
|
||||||
this.activePlugin = activePlugin;
|
this.activePlugin = activePlugin;
|
||||||
this.databaseHelper = databaseHelper;
|
this.databaseHelper = databaseHelper;
|
||||||
this.sp = sp;
|
|
||||||
this.omnipodAlertUtil = omnipodAlertUtil;
|
this.omnipodAlertUtil = omnipodAlertUtil;
|
||||||
|
|
||||||
delegate = new OmnipodManager(aapsLogger, sp, communicationService, podStateManager);
|
delegate = new OmnipodManager(aapsLogger, sp, communicationService, podStateManager);
|
||||||
|
@ -148,12 +148,12 @@ public class AapsOmnipodManager {
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Disposable disposable = delegate.pairAndPrime().subscribe(res -> //
|
executeCommand(() -> delegate.pairAndPrime().subscribe(res -> //
|
||||||
handleSetupActionResult(podInitActionType, podInitReceiver, res, System.currentTimeMillis(), null));
|
handleSetupActionResult(podInitActionType, podInitReceiver, res, System.currentTimeMillis(), null)));
|
||||||
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
podInitReceiver.returnInitTaskStatus(podInitActionType, false, comment);
|
podInitReceiver.returnInitTaskStatus(podInitActionType, false, comment);
|
||||||
addFailureToHistory(System.currentTimeMillis(), PodHistoryEntryType.PAIR_AND_PRIME, comment);
|
addFailureToHistory(System.currentTimeMillis(), PodHistoryEntryType.PAIR_AND_PRIME, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
|
@ -180,66 +180,68 @@ public class AapsOmnipodManager {
|
||||||
throw new CommandInitializationException("Basal profile mapping failed", ex);
|
throw new CommandInitializationException("Basal profile mapping failed", ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
Disposable disposable = delegate.insertCannula(basalSchedule, omnipodAlertUtil.getExpirationReminderTimeBeforeShutdown(), omnipodAlertUtil.getLowReservoirAlertUnits()).subscribe(res -> //
|
executeCommand(() -> delegate.insertCannula(basalSchedule, omnipodAlertUtil.getExpirationReminderTimeBeforeShutdown(), omnipodAlertUtil.getLowReservoirAlertUnits()).subscribe(res -> //
|
||||||
handleSetupActionResult(podInitActionType, podInitReceiver, res, System.currentTimeMillis(), profile));
|
handleSetupActionResult(podInitActionType, podInitReceiver, res, System.currentTimeMillis(), profile)));
|
||||||
|
} catch (Exception ex) {
|
||||||
|
String comment = translateException(ex);
|
||||||
|
podInitReceiver.returnInitTaskStatus(podInitActionType, false, comment);
|
||||||
|
addFailureToHistory(PodHistoryEntryType.FILL_CANNULA_SET_BASAL_PROFILE, comment);
|
||||||
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
|
}
|
||||||
|
|
||||||
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_NOT_ATTACHED));
|
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_NOT_ATTACHED));
|
||||||
|
|
||||||
cancelSuspendedFakeTbrIfExists();
|
cancelSuspendedFakeTbrIfExists();
|
||||||
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
} catch (Exception ex) {
|
|
||||||
String comment = handleAndTranslateException(ex);
|
|
||||||
podInitReceiver.returnInitTaskStatus(podInitActionType, false, comment);
|
|
||||||
addFailureToHistory(PodHistoryEntryType.FILL_CANNULA_SET_BASAL_PROFILE, comment);
|
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public PumpEnactResult configureAlerts(List<AlertConfiguration> alertConfigurations) {
|
public PumpEnactResult configureAlerts(List<AlertConfiguration> alertConfigurations) {
|
||||||
try {
|
try {
|
||||||
StatusResponse statusResponse = delegate.configureAlerts(alertConfigurations);
|
executeCommand(() -> delegate.configureAlerts(alertConfigurations));
|
||||||
addSuccessToHistory(PodHistoryEntryType.CONFIGURE_ALERTS, alertConfigurations);
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(false);
|
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(PodHistoryEntryType.CONFIGURE_ALERTS, comment);
|
addFailureToHistory(PodHistoryEntryType.CONFIGURE_ALERTS, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
addSuccessToHistory(PodHistoryEntryType.CONFIGURE_ALERTS, alertConfigurations);
|
||||||
|
return new PumpEnactResult(injector).success(true).enacted(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PumpEnactResult getPodStatus() {
|
public PumpEnactResult getPodStatus() {
|
||||||
|
StatusResponse statusResponse;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
StatusResponse statusResponse = delegate.getPodStatus();
|
statusResponse = executeCommand(delegate::getPodStatus);
|
||||||
addSuccessToHistory(PodHistoryEntryType.GET_POD_STATUS, statusResponse);
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(false);
|
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(PodHistoryEntryType.GET_POD_STATUS, comment);
|
addFailureToHistory(PodHistoryEntryType.GET_POD_STATUS, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
addSuccessToHistory(PodHistoryEntryType.GET_POD_STATUS, statusResponse);
|
||||||
|
return new PumpEnactResult(injector).success(true).enacted(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PumpEnactResult deactivatePod(PodInitReceiver podInitReceiver) {
|
public PumpEnactResult deactivatePod(PodInitReceiver podInitReceiver) {
|
||||||
try {
|
try {
|
||||||
delegate.deactivatePod();
|
executeCommand(delegate::deactivatePod);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
podInitReceiver.returnInitTaskStatus(PodInitActionType.DEACTIVATE_POD_WIZARD_STEP, false, comment);
|
podInitReceiver.returnInitTaskStatus(PodInitActionType.DEACTIVATE_POD_WIZARD_STEP, false, comment);
|
||||||
addFailureToHistory(PodHistoryEntryType.DEACTIVATE_POD, comment);
|
addFailureToHistory(PodHistoryEntryType.DEACTIVATE_POD, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
addSuccessToHistory(PodHistoryEntryType.DEACTIVATE_POD, null);
|
addSuccessToHistory(PodHistoryEntryType.DEACTIVATE_POD, null);
|
||||||
|
|
||||||
createSuspendedFakeTbrIfNotExists();
|
createSuspendedFakeTbrIfNotExists();
|
||||||
|
|
||||||
podInitReceiver.returnInitTaskStatus(PodInitActionType.DEACTIVATE_POD_WIZARD_STEP, true, null);
|
podInitReceiver.returnInitTaskStatus(PodInitActionType.DEACTIVATE_POD_WIZARD_STEP, true, null);
|
||||||
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PumpEnactResult setBasalProfile(Profile profile) {
|
public PumpEnactResult setBasalProfile(Profile profile, boolean showNotifications) {
|
||||||
if (profile == null) {
|
if (profile == null) {
|
||||||
String comment = getStringResource(R.string.omnipod_error_failed_to_set_profile_empty_profile);
|
String comment = getStringResource(R.string.omnipod_error_failed_to_set_profile_empty_profile);
|
||||||
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
@ -255,16 +257,13 @@ public class AapsOmnipodManager {
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
throw new CommandInitializationException("Basal profile mapping failed", ex);
|
throw new CommandInitializationException("Basal profile mapping failed", ex);
|
||||||
}
|
}
|
||||||
delegate.setBasalSchedule(basalSchedule, isBasalBeepsEnabled());
|
executeCommand(() -> delegate.setBasalSchedule(basalSchedule, isBasalBeepsEnabled()));
|
||||||
|
|
||||||
if (historyEntryType == PodHistoryEntryType.RESUME_DELIVERY) {
|
|
||||||
cancelSuspendedFakeTbrIfExists();
|
|
||||||
}
|
|
||||||
addSuccessToHistory(historyEntryType, profile.getBasalValues());
|
|
||||||
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
||||||
createSuspendedFakeTbrIfNotExists();
|
createSuspendedFakeTbrIfNotExists();
|
||||||
String comment = getStringResource(R.string.omnipod_error_set_basal_failed_delivery_suspended);
|
String comment = getStringResource(R.string.omnipod_error_set_basal_failed_delivery_suspended);
|
||||||
|
if (showNotifications) {
|
||||||
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
}
|
||||||
addFailureToHistory(historyEntryType, comment);
|
addFailureToHistory(historyEntryType, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
} catch (DeliveryStatusVerificationFailedException ex) {
|
} catch (DeliveryStatusVerificationFailedException ex) {
|
||||||
|
@ -276,20 +275,29 @@ public class AapsOmnipodManager {
|
||||||
// Happened when setting the new profile (after suspending delivery)
|
// Happened when setting the new profile (after suspending delivery)
|
||||||
comment = getStringResource(R.string.omnipod_error_set_basal_might_have_failed_delivery_might_be_suspended);
|
comment = getStringResource(R.string.omnipod_error_set_basal_might_have_failed_delivery_might_be_suspended);
|
||||||
}
|
}
|
||||||
|
if (showNotifications) {
|
||||||
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
}
|
||||||
addFailureToHistory(historyEntryType, comment);
|
addFailureToHistory(historyEntryType, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
|
if (showNotifications) {
|
||||||
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(Notification.FAILED_UDPATE_PROFILE, comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
}
|
||||||
addFailureToHistory(historyEntryType, comment);
|
addFailureToHistory(historyEntryType, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (historyEntryType == PodHistoryEntryType.RESUME_DELIVERY) {
|
||||||
|
cancelSuspendedFakeTbrIfExists();
|
||||||
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_SUSPENDED));
|
rxBus.send(new EventDismissNotification(Notification.OMNIPOD_POD_SUSPENDED));
|
||||||
showNotification(Notification.PROFILE_SET_OK,
|
}
|
||||||
resourceHelper.gs(R.string.profile_set_ok),
|
addSuccessToHistory(historyEntryType, profile.getBasalValues());
|
||||||
Notification.INFO, null);
|
|
||||||
|
if (showNotifications) {
|
||||||
|
showNotification(Notification.PROFILE_SET_OK, resourceHelper.gs(R.string.profile_set_ok), Notification.INFO, null);
|
||||||
|
}
|
||||||
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
}
|
}
|
||||||
|
@ -311,17 +319,17 @@ public class AapsOmnipodManager {
|
||||||
|
|
||||||
Date bolusStarted;
|
Date bolusStarted;
|
||||||
try {
|
try {
|
||||||
bolusCommandResult = delegate.bolus(PumpType.Insulet_Omnipod.determineCorrectBolusSize(detailedBolusInfo.insulin), beepsEnabled, beepsEnabled, detailedBolusInfo.isSMB ? null :
|
bolusCommandResult = executeCommand(() -> delegate.bolus(PumpType.Insulet_Omnipod.determineCorrectBolusSize(detailedBolusInfo.insulin), beepsEnabled, beepsEnabled, detailedBolusInfo.isSMB ? null :
|
||||||
(estimatedUnitsDelivered, percentage) -> {
|
(estimatedUnitsDelivered, percentage) -> {
|
||||||
EventOverviewBolusProgress progressUpdateEvent = EventOverviewBolusProgress.INSTANCE;
|
EventOverviewBolusProgress progressUpdateEvent = EventOverviewBolusProgress.INSTANCE;
|
||||||
progressUpdateEvent.setStatus(getStringResource(R.string.bolusdelivering, detailedBolusInfo.insulin));
|
progressUpdateEvent.setStatus(getStringResource(R.string.bolusdelivering, detailedBolusInfo.insulin));
|
||||||
progressUpdateEvent.setPercent(percentage);
|
progressUpdateEvent.setPercent(percentage);
|
||||||
sendEvent(progressUpdateEvent);
|
sendEvent(progressUpdateEvent);
|
||||||
});
|
}));
|
||||||
|
|
||||||
bolusStarted = new Date();
|
bolusStarted = new Date();
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(System.currentTimeMillis(), PodHistoryEntryType.SET_BOLUS, comment);
|
addFailureToHistory(System.currentTimeMillis(), PodHistoryEntryType.SET_BOLUS, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
@ -400,19 +408,19 @@ public class AapsOmnipodManager {
|
||||||
String comment = null;
|
String comment = null;
|
||||||
for (int i = 1; delegate.hasActiveBolus(); i++) {
|
for (int i = 1; delegate.hasActiveBolus(); i++) {
|
||||||
aapsLogger.debug(LTag.PUMP, "Attempting to cancel bolus (#{})", i);
|
aapsLogger.debug(LTag.PUMP, "Attempting to cancel bolus (#{})", i);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
delegate.cancelBolus(isBolusBeepsEnabled());
|
executeCommand(() -> delegate.cancelBolus(isBolusBeepsEnabled()));
|
||||||
aapsLogger.debug(LTag.PUMP, "Successfully cancelled bolus", i);
|
aapsLogger.debug(LTag.PUMP, "Successfully cancelled bolus", i);
|
||||||
addSuccessToHistory(System.currentTimeMillis(), PodHistoryEntryType.CANCEL_BOLUS, null);
|
addSuccessToHistory(System.currentTimeMillis(), PodHistoryEntryType.CANCEL_BOLUS, null);
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
} catch (PodFaultException ex) {
|
} catch (PodFaultException ex) {
|
||||||
aapsLogger.debug(LTag.PUMP, "Successfully cancelled bolus (implicitly because of a Pod Fault)");
|
aapsLogger.debug(LTag.PUMP, "Successfully cancelled bolus (implicitly because of a Pod Fault)");
|
||||||
showPodFaultNotification(ex.getFaultEvent().getFaultEventCode());
|
|
||||||
addSuccessToHistory(System.currentTimeMillis(), PodHistoryEntryType.CANCEL_BOLUS, null);
|
addSuccessToHistory(System.currentTimeMillis(), PodHistoryEntryType.CANCEL_BOLUS, null);
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
aapsLogger.debug(LTag.PUMP, "Failed to cancel bolus", ex);
|
aapsLogger.debug(LTag.PUMP, "Failed to cancel bolus", ex);
|
||||||
comment = handleAndTranslateException(ex);
|
comment = translateException(ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -423,7 +431,7 @@ public class AapsOmnipodManager {
|
||||||
public PumpEnactResult setTemporaryBasal(TempBasalPair tempBasalPair) {
|
public PumpEnactResult setTemporaryBasal(TempBasalPair tempBasalPair) {
|
||||||
boolean beepsEnabled = isTbrBeepsEnabled();
|
boolean beepsEnabled = isTbrBeepsEnabled();
|
||||||
try {
|
try {
|
||||||
delegate.setTemporaryBasal(PumpType.Insulet_Omnipod.determineCorrectBasalSize(tempBasalPair.getInsulinRate()), Duration.standardMinutes(tempBasalPair.getDurationMinutes()), beepsEnabled, beepsEnabled);
|
executeCommand(() -> delegate.setTemporaryBasal(PumpType.Insulet_Omnipod.determineCorrectBasalSize(tempBasalPair.getInsulinRate()), Duration.standardMinutes(tempBasalPair.getDurationMinutes()), beepsEnabled, beepsEnabled));
|
||||||
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
||||||
String comment = getStringResource(R.string.omnipod_cancelled_old_tbr_failed_to_set_new);
|
String comment = getStringResource(R.string.omnipod_cancelled_old_tbr_failed_to_set_new);
|
||||||
addFailureToHistory(PodHistoryEntryType.SET_TEMPORARY_BASAL, comment);
|
addFailureToHistory(PodHistoryEntryType.SET_TEMPORARY_BASAL, comment);
|
||||||
|
@ -453,7 +461,7 @@ public class AapsOmnipodManager {
|
||||||
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(PodHistoryEntryType.SET_TEMPORARY_BASAL, comment);
|
addFailureToHistory(PodHistoryEntryType.SET_TEMPORARY_BASAL, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
@ -467,14 +475,14 @@ public class AapsOmnipodManager {
|
||||||
|
|
||||||
public PumpEnactResult cancelTemporaryBasal() {
|
public PumpEnactResult cancelTemporaryBasal() {
|
||||||
try {
|
try {
|
||||||
delegate.cancelTemporaryBasal(isTbrBeepsEnabled());
|
executeCommand(() -> delegate.cancelTemporaryBasal(isTbrBeepsEnabled()));
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment;
|
String comment;
|
||||||
if (ex instanceof OmnipodException && !((OmnipodException) ex).isCertainFailure()) {
|
if (ex instanceof OmnipodException && !((OmnipodException) ex).isCertainFailure()) {
|
||||||
comment = getStringResource(R.string.omnipod_error_cancel_temp_basal_failed_uncertain);
|
comment = getStringResource(R.string.omnipod_error_cancel_temp_basal_failed_uncertain);
|
||||||
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
||||||
} else {
|
} else {
|
||||||
comment = handleAndTranslateException(ex);
|
comment = translateException(ex);
|
||||||
}
|
}
|
||||||
addFailureToHistory(PodHistoryEntryType.CANCEL_TEMPORARY_BASAL, comment);
|
addFailureToHistory(PodHistoryEntryType.CANCEL_TEMPORARY_BASAL, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
|
@ -495,59 +503,68 @@ public class AapsOmnipodManager {
|
||||||
|
|
||||||
public PumpEnactResult acknowledgeAlerts() {
|
public PumpEnactResult acknowledgeAlerts() {
|
||||||
try {
|
try {
|
||||||
delegate.acknowledgeAlerts();
|
executeCommand(delegate::acknowledgeAlerts);
|
||||||
addSuccessToHistory(PodHistoryEntryType.ACKNOWLEDGE_ALERTS, null);
|
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(PodHistoryEntryType.ACKNOWLEDGE_ALERTS, comment);
|
addFailureToHistory(PodHistoryEntryType.ACKNOWLEDGE_ALERTS, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
addSuccessToHistory(PodHistoryEntryType.ACKNOWLEDGE_ALERTS, null);
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PumpEnactResult suspendDelivery() {
|
public PumpEnactResult suspendDelivery() {
|
||||||
try {
|
try {
|
||||||
delegate.suspendDelivery(isBasalBeepsEnabled());
|
executeCommand(() -> delegate.suspendDelivery(isBasalBeepsEnabled()));
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment;
|
||||||
|
if (ex instanceof OmnipodException && !((OmnipodException) ex).isCertainFailure()) {
|
||||||
|
comment = getStringResource(R.string.omnipod_error_suspend_delivery_failed_uncertain);
|
||||||
|
} else {
|
||||||
|
comment = getStringResource(R.string.omnipod_error_suspend_delivery_failed);
|
||||||
|
}
|
||||||
addFailureToHistory(PodHistoryEntryType.SUSPEND_DELIVERY, comment);
|
addFailureToHistory(PodHistoryEntryType.SUSPEND_DELIVERY, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
addSuccessToHistory(PodHistoryEntryType.SUSPEND_DELIVERY, null);
|
addSuccessToHistory(PodHistoryEntryType.SUSPEND_DELIVERY, null);
|
||||||
|
|
||||||
createSuspendedFakeTbrIfNotExists();
|
createSuspendedFakeTbrIfNotExists();
|
||||||
|
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Updates the pods current time based on the device timezone and the pod's time zone
|
// Updates the pods current time based on the device timezone and the pod's time zone
|
||||||
public PumpEnactResult setTime() {
|
public PumpEnactResult setTime(boolean showNotifications) {
|
||||||
try {
|
try {
|
||||||
delegate.setTime(isBasalBeepsEnabled());
|
executeCommand(() -> delegate.setTime(isBasalBeepsEnabled()));
|
||||||
addSuccessToHistory(PodHistoryEntryType.SET_TIME, null);
|
|
||||||
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
} catch (CommandFailedAfterChangingDeliveryStatusException ex) {
|
||||||
createSuspendedFakeTbrIfNotExists();
|
createSuspendedFakeTbrIfNotExists();
|
||||||
String comment = getStringResource(R.string.omnipod_error_set_time_failed_delivery_suspended);
|
String comment = getStringResource(R.string.omnipod_error_set_time_failed_delivery_suspended);
|
||||||
|
if (showNotifications) {
|
||||||
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
}
|
||||||
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
} catch (DeliveryStatusVerificationFailedException ex) {
|
} catch (DeliveryStatusVerificationFailedException ex) {
|
||||||
String comment = getStringResource(R.string.omnipod_error_set_time_failed_delivery_might_be_suspended);
|
String comment = getStringResource(R.string.omnipod_error_set_time_failed_delivery_might_be_suspended);
|
||||||
|
if (showNotifications) {
|
||||||
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
showNotification(comment, Notification.URGENT, R.raw.boluserror);
|
||||||
|
}
|
||||||
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
} catch (Exception ex) {
|
} catch (Exception ex) {
|
||||||
String comment = handleAndTranslateException(ex);
|
String comment = translateException(ex);
|
||||||
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
addFailureToHistory(PodHistoryEntryType.SET_TIME, comment);
|
||||||
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
return new PumpEnactResult(injector).success(false).enacted(false).comment(comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
addSuccessToHistory(PodHistoryEntryType.SET_TIME, null);
|
||||||
return new PumpEnactResult(injector).success(true).enacted(true);
|
return new PumpEnactResult(injector).success(true).enacted(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public PodInfoRecentPulseLog readPulseLog() {
|
public PodInfoRecentPulseLog readPulseLog() {
|
||||||
PodInfoResponse response = delegate.getPodInfo(PodInfoType.RECENT_PULSE_LOG);
|
PodInfoResponse response = executeCommand(() -> delegate.getPodInfo(PodInfoType.RECENT_PULSE_LOG));
|
||||||
return (PodInfoRecentPulseLog) response.getPodInfo();
|
return (PodInfoRecentPulseLog) response.getPodInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -734,7 +751,37 @@ public class AapsOmnipodManager {
|
||||||
podInitReceiver.returnInitTaskStatus(podInitActionType, res.getResultType().isSuccess(), comment);
|
podInitReceiver.returnInitTaskStatus(podInitActionType, res.getResultType().isSuccess(), comment);
|
||||||
}
|
}
|
||||||
|
|
||||||
private String handleAndTranslateException(Exception ex) {
|
private void executeCommand(Runnable runnable) {
|
||||||
|
try {
|
||||||
|
runnable.run();
|
||||||
|
} catch (Exception ex) {
|
||||||
|
handleException(ex);
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private <T> T executeCommand(Supplier<T> supplier) {
|
||||||
|
try {
|
||||||
|
return supplier.get();
|
||||||
|
} catch (Exception ex) {
|
||||||
|
handleException(ex);
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleException(Exception ex) {
|
||||||
|
if (ex instanceof OmnipodException) {
|
||||||
|
aapsLogger.error(LTag.PUMP, String.format("Caught OmnipodException[certainFailure=%s] from OmnipodManager", ((OmnipodException) ex).isCertainFailure()), ex);
|
||||||
|
if (ex instanceof PodFaultException) {
|
||||||
|
FaultEventCode faultEventCode = ((PodFaultException) ex).getFaultEvent().getFaultEventCode();
|
||||||
|
showPodFaultNotification(faultEventCode);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
aapsLogger.error(LTag.PUMP, "Caught an unexpected non-OmnipodException from OmnipodManager", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String translateException(Exception ex) {
|
||||||
String comment;
|
String comment;
|
||||||
|
|
||||||
if (ex instanceof OmnipodException) {
|
if (ex instanceof OmnipodException) {
|
||||||
|
@ -770,18 +817,15 @@ public class AapsOmnipodManager {
|
||||||
comment = getStringResource(R.string.omnipod_driver_error_not_enough_data);
|
comment = getStringResource(R.string.omnipod_driver_error_not_enough_data);
|
||||||
} else if (ex instanceof PodFaultException) {
|
} else if (ex instanceof PodFaultException) {
|
||||||
FaultEventCode faultEventCode = ((PodFaultException) ex).getFaultEvent().getFaultEventCode();
|
FaultEventCode faultEventCode = ((PodFaultException) ex).getFaultEvent().getFaultEventCode();
|
||||||
showPodFaultNotification(faultEventCode);
|
|
||||||
comment = createPodFaultErrorMessage(faultEventCode);
|
comment = createPodFaultErrorMessage(faultEventCode);
|
||||||
} else if (ex instanceof PodReturnedErrorResponseException) {
|
} else if (ex instanceof PodReturnedErrorResponseException) {
|
||||||
comment = getStringResource(R.string.omnipod_driver_error_pod_returned_error_response);
|
comment = getStringResource(R.string.omnipod_driver_error_pod_returned_error_response);
|
||||||
} else {
|
} else {
|
||||||
// Shouldn't be reachable
|
// Shouldn't be reachable
|
||||||
comment = getStringResource(R.string.omnipod_driver_error_unexpected_exception_type, ex.getClass().getName());
|
comment = getStringResource(R.string.omnipod_driver_error_unexpected_exception_type, ex.getClass().getName(), ex.getMessage());
|
||||||
}
|
}
|
||||||
aapsLogger.error(LTag.PUMP, String.format("Caught OmnipodException[certainFailure=%s] from OmnipodManager (user-friendly error message: %s)", ((OmnipodException) ex).isCertainFailure(), comment), ex);
|
|
||||||
} else {
|
} else {
|
||||||
comment = getStringResource(R.string.omnipod_driver_error_unexpected_exception_type, ex.getClass().getName(), ex.getMessage());
|
comment = getStringResource(R.string.omnipod_driver_error_unexpected_exception_type, ex.getClass().getName(), ex.getMessage());
|
||||||
aapsLogger.error(LTag.PUMP, String.format("Caught unexpected exception type[certainFailure=false] from OmnipodManager (user-friendly error message: %s)", comment), ex);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return comment;
|
return comment;
|
||||||
|
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandAcknowledgeAlerts extends OmnipodCustomCommand {
|
||||||
|
public CommandAcknowledgeAlerts() {
|
||||||
|
super(OmnipodCustomCommandType.ACKNOWLEDGE_ALERTS);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandGetPodStatus extends OmnipodCustomCommand {
|
||||||
|
public CommandGetPodStatus() {
|
||||||
|
super(OmnipodCustomCommandType.GET_POD_STATUS);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandHandleTimeChange extends OmnipodCustomCommand {
|
||||||
|
private final boolean requestedByUser;
|
||||||
|
|
||||||
|
public CommandHandleTimeChange(boolean requestedByUser) {
|
||||||
|
super(OmnipodCustomCommandType.HANDLE_TIME_CHANGE);
|
||||||
|
this.requestedByUser = requestedByUser;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isRequestedByUser() {
|
||||||
|
return requestedByUser;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandReadPulseLog extends OmnipodCustomCommand {
|
||||||
|
public CommandReadPulseLog() {
|
||||||
|
super(OmnipodCustomCommandType.READ_PULSE_LOG);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandResumeDelivery extends OmnipodCustomCommand {
|
||||||
|
public CommandResumeDelivery() {
|
||||||
|
super(OmnipodCustomCommandType.RESUME_DELIVERY);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandSuspendDelivery extends OmnipodCustomCommand {
|
||||||
|
public CommandSuspendDelivery() {
|
||||||
|
super(OmnipodCustomCommandType.SUSPEND_DELIVERY);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public final class CommandUpdateAlertConfiguration extends OmnipodCustomCommand {
|
||||||
|
public CommandUpdateAlertConfiguration() {
|
||||||
|
super(OmnipodCustomCommandType.UPDATE_ALERT_CONFIGURATION);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,21 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import info.nightscout.androidaps.plugins.general.command.defs.CustomCommand;
|
||||||
|
|
||||||
|
public abstract class OmnipodCustomCommand implements CustomCommand {
|
||||||
|
private final OmnipodCustomCommandType type;
|
||||||
|
|
||||||
|
OmnipodCustomCommand(@NonNull OmnipodCustomCommandType type) {
|
||||||
|
this.type = type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public final OmnipodCustomCommandType getType() {
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public final String getStatusDescription() {
|
||||||
|
return type.getDescription();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,21 @@
|
||||||
|
package info.nightscout.androidaps.plugins.pump.omnipod.queue.command;
|
||||||
|
|
||||||
|
public enum OmnipodCustomCommandType {
|
||||||
|
ACKNOWLEDGE_ALERTS("ACKNOWLEDGE ALERTS"),
|
||||||
|
GET_POD_STATUS("GET POD STATUS"),
|
||||||
|
READ_PULSE_LOG("READ PULSE LOG"),
|
||||||
|
SUSPEND_DELIVERY("SUSPEND DELIVERY"),
|
||||||
|
RESUME_DELIVERY("RESUME DELIVERY"),
|
||||||
|
HANDLE_TIME_CHANGE("HANDLE TIME CHANGE"),
|
||||||
|
UPDATE_ALERT_CONFIGURATION("UPDATE ALERT CONFIGURATION");
|
||||||
|
|
||||||
|
private final String description;
|
||||||
|
|
||||||
|
OmnipodCustomCommandType(String description) {
|
||||||
|
this.description = description;
|
||||||
|
}
|
||||||
|
|
||||||
|
String getDescription() {
|
||||||
|
return description;
|
||||||
|
}
|
||||||
|
}
|
|
@ -194,13 +194,13 @@ public class OmnipodRileyLinkCommunicationManager extends RileyLinkCommunication
|
||||||
// receive an ACK instead of a normal response, or a partial response and a communication timeout
|
// receive an ACK instead of a normal response, or a partial response and a communication timeout
|
||||||
if (message.isNonceResyncable() && !message.containsBlock(DeactivatePodCommand.class)) {
|
if (message.isNonceResyncable() && !message.containsBlock(DeactivatePodCommand.class)) {
|
||||||
OmnipodMessage paddedMessage = new OmnipodMessage(message);
|
OmnipodMessage paddedMessage = new OmnipodMessage(message);
|
||||||
// If messages are nonce resyncable, we want do distinguish between certain and uncertain failures for verification purposes
|
// If messages are nonce resyncable, we want to distinguish between certain and uncertain failures for verification purposes
|
||||||
// However, some commands (e.g. cancel delivery) are single packet command by nature. When we get a timeout with a single packet,
|
// However, some commands (e.g. cancel delivery) are single packet command by nature. When we get a timeout with a single packet,
|
||||||
// we are unsure whether or not the command was received by the pod
|
// we are unsure whether or not the command was received by the pod
|
||||||
// However, if we send > 1 packet, we know that the command wasn't received if we never send the subsequent packets,
|
// However, if we send > 1 packet, we know that the command wasn't received if we never send the subsequent packets,
|
||||||
// because the last packet contains the CRC.
|
// because the last packet contains the CRC.
|
||||||
// So we pad the message with get status commands to make it > packet
|
// So we pad the message with get status commands to make it > packet
|
||||||
paddedMessage.padWithGetStatusCommands(PacketType.PDM.getMaxBodyLength()); // First packet is of type PDM
|
paddedMessage.padWithGetStatusCommands(PacketType.PDM.getMaxBodyLength(), aapsLogger); // First packet is of type PDM
|
||||||
encodedMessage = paddedMessage.getEncoded();
|
encodedMessage = paddedMessage.getEncoded();
|
||||||
} else {
|
} else {
|
||||||
encodedMessage = message.getEncoded();
|
encodedMessage = message.getEncoded();
|
||||||
|
|
|
@ -11,6 +11,7 @@ import android.view.View
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import dagger.android.support.DaggerFragment
|
import dagger.android.support.DaggerFragment
|
||||||
import info.nightscout.androidaps.Constants
|
import info.nightscout.androidaps.Constants
|
||||||
|
import info.nightscout.androidaps.activities.ErrorHelperActivity
|
||||||
import info.nightscout.androidaps.events.EventPreferenceChange
|
import info.nightscout.androidaps.events.EventPreferenceChange
|
||||||
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
import info.nightscout.androidaps.interfaces.ActivePluginProvider
|
||||||
import info.nightscout.androidaps.interfaces.CommandQueueProvider
|
import info.nightscout.androidaps.interfaces.CommandQueueProvider
|
||||||
|
@ -22,15 +23,14 @@ import info.nightscout.androidaps.plugins.pump.common.hw.rileylink.dialog.RileyL
|
||||||
import info.nightscout.androidaps.plugins.pump.common.hw.rileylink.service.RileyLinkServiceData
|
import info.nightscout.androidaps.plugins.pump.common.hw.rileylink.service.RileyLinkServiceData
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.OmnipodPumpPlugin
|
import info.nightscout.androidaps.plugins.pump.omnipod.OmnipodPumpPlugin
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.R
|
import info.nightscout.androidaps.plugins.pump.omnipod.R
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodCommandType
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.definition.OmnipodStatusRequestType
|
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.OmnipodConstants
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.OmnipodConstants
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.PodProgressStatus
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.definition.PodProgressStatus
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.driver.manager.PodStateManager
|
import info.nightscout.androidaps.plugins.pump.omnipod.driver.manager.PodStateManager
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodPumpValuesChanged
|
import info.nightscout.androidaps.plugins.pump.omnipod.event.EventOmnipodPumpValuesChanged
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.manager.AapsOmnipodManager
|
import info.nightscout.androidaps.plugins.pump.omnipod.manager.AapsOmnipodManager
|
||||||
|
import info.nightscout.androidaps.plugins.pump.omnipod.queue.command.*
|
||||||
import info.nightscout.androidaps.plugins.pump.omnipod.util.AapsOmnipodUtil
|
import info.nightscout.androidaps.plugins.pump.omnipod.util.AapsOmnipodUtil
|
||||||
import info.nightscout.androidaps.queue.commands.Command
|
import info.nightscout.androidaps.queue.Callback
|
||||||
import info.nightscout.androidaps.queue.events.EventQueueChanged
|
import info.nightscout.androidaps.queue.events.EventQueueChanged
|
||||||
import info.nightscout.androidaps.utils.DateUtil
|
import info.nightscout.androidaps.utils.DateUtil
|
||||||
import info.nightscout.androidaps.utils.FabricPrivacy
|
import info.nightscout.androidaps.utils.FabricPrivacy
|
||||||
|
@ -92,11 +92,6 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
||||||
super.onViewCreated(view, savedInstanceState)
|
super.onViewCreated(view, savedInstanceState)
|
||||||
|
|
||||||
omnipod_button_resume_delivery.setOnClickListener {
|
|
||||||
disablePodActionButtons()
|
|
||||||
commandQueue.startPump(null)
|
|
||||||
}
|
|
||||||
|
|
||||||
omnipod_button_pod_mgmt.setOnClickListener {
|
omnipod_button_pod_mgmt.setOnClickListener {
|
||||||
if (omnipodPumpPlugin.rileyLinkService?.verifyConfiguration() == true) {
|
if (omnipodPumpPlugin.rileyLinkService?.verifyConfiguration() == true) {
|
||||||
activity?.let { activity ->
|
activity?.let { activity ->
|
||||||
|
@ -110,10 +105,16 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
omnipod_button_resume_delivery.setOnClickListener {
|
||||||
|
disablePodActionButtons()
|
||||||
|
commandQueue.customCommand(CommandResumeDelivery(),
|
||||||
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_resume_delivery), true).messageOnSuccess(resourceHelper.gs(R.string.omnipod_delivery_resumed)))
|
||||||
|
}
|
||||||
|
|
||||||
omnipod_button_refresh_status.setOnClickListener {
|
omnipod_button_refresh_status.setOnClickListener {
|
||||||
disablePodActionButtons()
|
disablePodActionButtons()
|
||||||
omnipodPumpPlugin.addPodStatusRequest(OmnipodStatusRequestType.GET_POD_STATE);
|
commandQueue.customCommand(CommandGetPodStatus(),
|
||||||
commandQueue.readStatus("Clicked Refresh", null)
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_refresh_status), false))
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_button_rileylink_stats.setOnClickListener {
|
omnipod_button_rileylink_stats.setOnClickListener {
|
||||||
|
@ -126,26 +127,29 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
|
|
||||||
omnipod_button_acknowledge_active_alerts.setOnClickListener {
|
omnipod_button_acknowledge_active_alerts.setOnClickListener {
|
||||||
disablePodActionButtons()
|
disablePodActionButtons()
|
||||||
omnipodPumpPlugin.addPodStatusRequest(OmnipodStatusRequestType.ACKNOWLEDGE_ALERTS);
|
commandQueue.customCommand(CommandAcknowledgeAlerts(),
|
||||||
commandQueue.readStatus("Clicked Acknowledge Alert", null)
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_acknowledge_alerts), false)
|
||||||
|
.messageOnSuccess(resourceHelper.gs(R.string.omnipod_acknowledged_alerts)))
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_button_suspend_delivery.setOnClickListener {
|
omnipod_button_suspend_delivery.setOnClickListener {
|
||||||
disablePodActionButtons()
|
disablePodActionButtons()
|
||||||
omnipodPumpPlugin.addPodStatusRequest(OmnipodStatusRequestType.SUSPEND_DELIVERY);
|
commandQueue.customCommand(CommandSuspendDelivery(),
|
||||||
commandQueue.readStatus("Clicked Suspend Delivery", null)
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_suspend_delivery), true)
|
||||||
|
.messageOnSuccess(resourceHelper.gs(R.string.omnipod_suspended_delivery)))
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_button_set_time.setOnClickListener {
|
omnipod_button_set_time.setOnClickListener {
|
||||||
disablePodActionButtons()
|
disablePodActionButtons()
|
||||||
omnipodPumpPlugin.addPodStatusRequest(OmnipodStatusRequestType.SET_TIME);
|
commandQueue.customCommand(CommandHandleTimeChange(true),
|
||||||
commandQueue.readStatus("Clicked Set Time", null)
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_set_time), true)
|
||||||
|
.messageOnSuccess(resourceHelper.gs(R.string.omnipod_time_on_pod_updated)))
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_button_pulse_log.setOnClickListener {
|
omnipod_button_pulse_log.setOnClickListener {
|
||||||
disablePodActionButtons()
|
disablePodActionButtons()
|
||||||
omnipodPumpPlugin.addPodStatusRequest(OmnipodStatusRequestType.GET_PULSE_LOG);
|
commandQueue.customCommand(CommandReadPulseLog(),
|
||||||
commandQueue.readStatus("Clicked Pulse Log", null)
|
ErrorDialogCallback(resourceHelper.gs(R.string.omnipod_failed_to_retrieve_pulse_log), false))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -219,7 +223,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
updateTempBasal()
|
updateTempBasal()
|
||||||
updatePodStatus()
|
updatePodStatus()
|
||||||
|
|
||||||
val errors = ArrayList<String>();
|
val errors = ArrayList<String>()
|
||||||
if (omnipodPumpPlugin.rileyLinkService != null) {
|
if (omnipodPumpPlugin.rileyLinkService != null) {
|
||||||
val rileyLinkErrorDescription = omnipodPumpPlugin.rileyLinkService.errorDescription
|
val rileyLinkErrorDescription = omnipodPumpPlugin.rileyLinkService.errorDescription
|
||||||
if (StringUtils.isNotEmpty(rileyLinkErrorDescription)) {
|
if (StringUtils.isNotEmpty(rileyLinkErrorDescription)) {
|
||||||
|
@ -285,7 +289,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
|
|
||||||
// total delivered
|
// total delivered
|
||||||
omnipod_total_delivered.text = if (podStateManager.isPodActivationCompleted && podStateManager.totalInsulinDelivered != null) {
|
omnipod_total_delivered.text = if (podStateManager.isPodActivationCompleted && podStateManager.totalInsulinDelivered != null) {
|
||||||
resourceHelper.gs(R.string.omnipod_total_delivered, podStateManager.totalInsulinDelivered - OmnipodConstants.POD_SETUP_UNITS);
|
resourceHelper.gs(R.string.omnipod_total_delivered, podStateManager.totalInsulinDelivered - OmnipodConstants.POD_SETUP_UNITS)
|
||||||
} else {
|
} else {
|
||||||
PLACEHOLDER
|
PLACEHOLDER
|
||||||
}
|
}
|
||||||
|
@ -386,7 +390,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
text += " (" + resourceHelper.gs(R.string.omnipod_uncertain) + ")"
|
text += " (" + resourceHelper.gs(R.string.omnipod_uncertain) + ")"
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_last_bolus.text = text;
|
omnipod_last_bolus.text = text
|
||||||
omnipod_last_bolus.setTextColor(textColor)
|
omnipod_last_bolus.setTextColor(textColor)
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
@ -399,9 +403,9 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
if (podStateManager.isPodActivationCompleted && podStateManager.isTempBasalRunning) {
|
if (podStateManager.isPodActivationCompleted && podStateManager.isTempBasalRunning) {
|
||||||
val now = DateTime.now()
|
val now = DateTime.now()
|
||||||
|
|
||||||
val startTime = podStateManager.tempBasalStartTime;
|
val startTime = podStateManager.tempBasalStartTime
|
||||||
val amount = podStateManager.tempBasalAmount
|
val amount = podStateManager.tempBasalAmount
|
||||||
val duration = podStateManager.tempBasalDuration;
|
val duration = podStateManager.tempBasalDuration
|
||||||
|
|
||||||
val minutesRunning = Duration(startTime, now).standardMinutes
|
val minutesRunning = Duration(startTime, now).standardMinutes
|
||||||
|
|
||||||
|
@ -415,7 +419,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
text += " (" + resourceHelper.gs(R.string.omnipod_uncertain) + ")"
|
text += " (" + resourceHelper.gs(R.string.omnipod_uncertain) + ")"
|
||||||
}
|
}
|
||||||
|
|
||||||
omnipod_temp_basal.text = text;
|
omnipod_temp_basal.text = text
|
||||||
omnipod_temp_basal.setTextColor(textColor)
|
omnipod_temp_basal.setTextColor(textColor)
|
||||||
} else {
|
} else {
|
||||||
omnipod_temp_basal.text = PLACEHOLDER
|
omnipod_temp_basal.text = PLACEHOLDER
|
||||||
|
@ -456,8 +460,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun updateResumeDeliveryButton() {
|
private fun updateResumeDeliveryButton() {
|
||||||
val queueEmptyOrStartingPump = isQueueEmpty() || commandQueue.isRunning(Command.CommandType.START_PUMP)
|
if (podStateManager.isPodRunning && (podStateManager.isSuspended || commandQueue.isCustomCommandInQueue(CommandResumeDelivery::class.java))) {
|
||||||
if (podStateManager.isPodActivationCompleted && podStateManager.isSuspended && queueEmptyOrStartingPump) {
|
|
||||||
omnipod_button_resume_delivery.visibility = View.VISIBLE
|
omnipod_button_resume_delivery.visibility = View.VISIBLE
|
||||||
omnipod_button_resume_delivery.isEnabled = rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
omnipod_button_resume_delivery.isEnabled = rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
||||||
} else {
|
} else {
|
||||||
|
@ -466,7 +469,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun updateAcknowledgeAlertsButton() {
|
private fun updateAcknowledgeAlertsButton() {
|
||||||
if (podStateManager.isPodRunning && podStateManager.hasActiveAlerts()) {
|
if (podStateManager.isPodRunning && (podStateManager.hasActiveAlerts() || commandQueue.isCustomCommandInQueue(CommandAcknowledgeAlerts::class.java))) {
|
||||||
omnipod_button_acknowledge_active_alerts.visibility = View.VISIBLE
|
omnipod_button_acknowledge_active_alerts.visibility = View.VISIBLE
|
||||||
omnipod_button_acknowledge_active_alerts.isEnabled = rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
omnipod_button_acknowledge_active_alerts.isEnabled = rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
||||||
} else {
|
} else {
|
||||||
|
@ -476,7 +479,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
|
|
||||||
private fun updateSuspendDeliveryButton() {
|
private fun updateSuspendDeliveryButton() {
|
||||||
// If the Pod is currently suspended, we show the Resume delivery button instead.
|
// If the Pod is currently suspended, we show the Resume delivery button instead.
|
||||||
if (omnipodManager.isSuspendDeliveryButtonEnabled && !(podStateManager.isPodRunning && podStateManager.isSuspended)) {
|
if (omnipodManager.isSuspendDeliveryButtonEnabled && podStateManager.isPodRunning && (!podStateManager.isSuspended || commandQueue.isCustomCommandInQueue(CommandSuspendDelivery::class.java))) {
|
||||||
omnipod_button_suspend_delivery.visibility = View.VISIBLE
|
omnipod_button_suspend_delivery.visibility = View.VISIBLE
|
||||||
omnipod_button_suspend_delivery.isEnabled = podStateManager.isPodRunning && !podStateManager.isSuspended && rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
omnipod_button_suspend_delivery.isEnabled = podStateManager.isPodRunning && !podStateManager.isSuspended && rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
||||||
} else {
|
} else {
|
||||||
|
@ -485,7 +488,7 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun updateSetTimeButton() {
|
private fun updateSetTimeButton() {
|
||||||
if (podStateManager.isPodRunning && (podStateManager.timeDeviatesMoreThan(Duration.standardMinutes(5)) || omnipodPumpPlugin.currentCommand == OmnipodCommandType.SET_TIME)) {
|
if (podStateManager.isPodRunning && (podStateManager.timeDeviatesMoreThan(Duration.standardMinutes(5)) || commandQueue.isCustomCommandInQueue(CommandHandleTimeChange::class.java))) {
|
||||||
omnipod_button_set_time.visibility = View.VISIBLE
|
omnipod_button_set_time.visibility = View.VISIBLE
|
||||||
omnipod_button_set_time.isEnabled = !podStateManager.isSuspended && rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
omnipod_button_set_time.isEnabled = !podStateManager.isSuspended && rileyLinkServiceData.rileyLinkServiceState.isReady && isQueueEmpty()
|
||||||
} else {
|
} else {
|
||||||
|
@ -504,8 +507,29 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
|
|
||||||
private fun displayNotConfiguredDialog() {
|
private fun displayNotConfiguredDialog() {
|
||||||
context?.let {
|
context?.let {
|
||||||
|
UIRunnable(Runnable {
|
||||||
OKDialog.show(it, resourceHelper.gs(R.string.omnipod_warning),
|
OKDialog.show(it, resourceHelper.gs(R.string.omnipod_warning),
|
||||||
resourceHelper.gs(R.string.omnipod_error_operation_not_possible_no_configuration), null)
|
resourceHelper.gs(R.string.omnipod_error_operation_not_possible_no_configuration), null)
|
||||||
|
}).run()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun displayErrorDialog(title: String, message: String, withSound: Boolean) {
|
||||||
|
context?.let {
|
||||||
|
val i = Intent(it, ErrorHelperActivity::class.java)
|
||||||
|
i.putExtra("soundid", if (withSound) R.raw.boluserror else 0)
|
||||||
|
i.putExtra("status", message)
|
||||||
|
i.putExtra("title", title)
|
||||||
|
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
|
||||||
|
it.startActivity(i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun displayOkDialog(title: String, message: String) {
|
||||||
|
context?.let {
|
||||||
|
UIRunnable(Runnable {
|
||||||
|
OKDialog.show(it, title, message, null)
|
||||||
|
}).run()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -568,4 +592,24 @@ class OmnipodFragment : DaggerFragment() {
|
||||||
return Duration.standardMinutes(sp.getInt(resourceHelper.gs(R.string.key_pump_unreachable_threshold_minutes), Constants.DEFAULT_PUMP_UNREACHABLE_THRESHOLD_MINUTES).toLong())
|
return Duration.standardMinutes(sp.getInt(resourceHelper.gs(R.string.key_pump_unreachable_threshold_minutes), Constants.DEFAULT_PUMP_UNREACHABLE_THRESHOLD_MINUTES).toLong())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inner class ErrorDialogCallback(private val errorMessagePrefix: String, private val withSoundOnError: Boolean) : Callback() {
|
||||||
|
private var messageOnSuccess: String? = null
|
||||||
|
|
||||||
|
override fun run() {
|
||||||
|
if (result.success) {
|
||||||
|
val messageOnSuccess = this.messageOnSuccess
|
||||||
|
if (messageOnSuccess != null) {
|
||||||
|
displayOkDialog(resourceHelper.gs(R.string.omnipod_confirmation), messageOnSuccess)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
displayErrorDialog(resourceHelper.gs(R.string.omnipod_warning), resourceHelper.gs(R.string.omnipod_two_strings_concatenated_by_colon, errorMessagePrefix, result.comment), withSoundOnError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun messageOnSuccess(message: String): ErrorDialogCallback {
|
||||||
|
messageOnSuccess = message
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -806,18 +806,6 @@
|
||||||
android:text="@string/omnipod_read_pulse_log_short"
|
android:text="@string/omnipod_read_pulse_log_short"
|
||||||
android:visibility="gone" />
|
android:visibility="gone" />
|
||||||
|
|
||||||
<Button
|
|
||||||
android:id="@+id/omnipod_button_suspend_delivery"
|
|
||||||
style="@style/ButtonSmallFontStyle"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="match_parent"
|
|
||||||
android:layout_weight="1"
|
|
||||||
android:drawableTop="@drawable/ic_loop_disabled"
|
|
||||||
android:paddingLeft="0dp"
|
|
||||||
android:paddingRight="0dp"
|
|
||||||
android:text="@string/omnipod_suspend_delivery_short"
|
|
||||||
android:visibility="gone" />
|
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
android:id="@+id/omnipod_button_set_time"
|
android:id="@+id/omnipod_button_set_time"
|
||||||
style="@style/ButtonSmallFontStyle"
|
style="@style/ButtonSmallFontStyle"
|
||||||
|
@ -842,6 +830,18 @@
|
||||||
android:text="@string/omnipod_resume_delivery"
|
android:text="@string/omnipod_resume_delivery"
|
||||||
android:visibility="gone" />
|
android:visibility="gone" />
|
||||||
|
|
||||||
|
<Button
|
||||||
|
android:id="@+id/omnipod_button_suspend_delivery"
|
||||||
|
style="@style/ButtonSmallFontStyle"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:layout_weight="1"
|
||||||
|
android:drawableTop="@drawable/ic_loop_disabled"
|
||||||
|
android:paddingLeft="0dp"
|
||||||
|
android:paddingRight="0dp"
|
||||||
|
android:text="@string/omnipod_suspend_delivery_short"
|
||||||
|
android:visibility="gone" />
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
||||||
</LinearLayout>
|
</LinearLayout>
|
||||||
|
|
|
@ -114,7 +114,7 @@
|
||||||
<string name="omnipod_init_pod_wizard_step1_title">Fill the Pod</string>
|
<string name="omnipod_init_pod_wizard_step1_title">Fill the Pod</string>
|
||||||
<string name="omnipod_init_pod_wizard_step1_desc">\nFill the new Pod with enough insulin for 3 days.\n\nListen for two beeps from the Pod during the filling process. These indicate that the minimum amount of 85U has been inserted. Be sure to completely empty the fill syringe, even after hearing the two beeps.\n\nAfter filling the Pod, please press <b>Next</b>.\n\n<b>Note:</b> do not remove the Pod\'s needle cap at this time.\n<b>Note:</b> please place the RileyLink in an upright position and place the Pod a few inches away.</string>
|
<string name="omnipod_init_pod_wizard_step1_desc">\nFill the new Pod with enough insulin for 3 days.\n\nListen for two beeps from the Pod during the filling process. These indicate that the minimum amount of 85U has been inserted. Be sure to completely empty the fill syringe, even after hearing the two beeps.\n\nAfter filling the Pod, please press <b>Next</b>.\n\n<b>Note:</b> do not remove the Pod\'s needle cap at this time.\n<b>Note:</b> please place the RileyLink in an upright position and place the Pod a few inches away.</string>
|
||||||
<string name="omnipod_init_pod_wizard_step2_title">Priming</string>
|
<string name="omnipod_init_pod_wizard_step2_title">Priming</string>
|
||||||
<string name="omnipod_init_pod_wizard_step2_action_header">Trying to pair with the new Pod and prime it.\n\nWhen all items are checked, you can press <b>Next</b>.\n\n<b>Note:</b> please place the RileyLink in an upright position and place the Pod a few inches away it.</string>
|
<string name="omnipod_init_pod_wizard_step2_action_header">Trying to pair with the new Pod and prime it.\n\nWhen all items are checked, you can press <b>Next</b>.\n\n<b>Note:</b> please place the RileyLink in an upright position and place the Pod a few inches away from it.</string>
|
||||||
<string name="omnipod_init_pod_wizard_step3_title">Attach the Pod</string>
|
<string name="omnipod_init_pod_wizard_step3_title">Attach the Pod</string>
|
||||||
<string name="omnipod_init_pod_wizard_step3_desc">\nPrepare the infusion site. Remove the Pod\'s needle cap and adhesive backing and attach the Pod to the infusion site.\n\nIf the cannula sticks out, please press <b>Cancel</b> and discard your Pod.\n\nPress <b>Next</b> to insert the cannula and begin basal delivery.</string>
|
<string name="omnipod_init_pod_wizard_step3_desc">\nPrepare the infusion site. Remove the Pod\'s needle cap and adhesive backing and attach the Pod to the infusion site.\n\nIf the cannula sticks out, please press <b>Cancel</b> and discard your Pod.\n\nPress <b>Next</b> to insert the cannula and begin basal delivery.</string>
|
||||||
<string name="omnipod_init_pod_wizard_step4_title">Inserting cannula</string>
|
<string name="omnipod_init_pod_wizard_step4_title">Inserting cannula</string>
|
||||||
|
@ -145,6 +145,8 @@
|
||||||
<string name="omnipod_error_set_basal_might_have_failed_delivery_might_be_suspended">Setting basal profile might have failed. Delivery might be suspended! Please manually refresh the Pod status from the Omnipod tab and resume delivery if needed.</string>
|
<string name="omnipod_error_set_basal_might_have_failed_delivery_might_be_suspended">Setting basal profile might have failed. Delivery might be suspended! Please manually refresh the Pod status from the Omnipod tab and resume delivery if needed.</string>
|
||||||
<string name="omnipod_error_set_basal_failed_delivery_suspended">Setting basal profile failed. Delivery is suspended! Please manually resume delivery from the Omnipod tab.</string>
|
<string name="omnipod_error_set_basal_failed_delivery_suspended">Setting basal profile failed. Delivery is suspended! Please manually resume delivery from the Omnipod tab.</string>
|
||||||
<string name="omnipod_error_cancel_temp_basal_failed_uncertain">Cancelling temp basal might have failed. Please manually refresh the Pod status from the Omnipod tab.</string>
|
<string name="omnipod_error_cancel_temp_basal_failed_uncertain">Cancelling temp basal might have failed. Please manually refresh the Pod status from the Omnipod tab.</string>
|
||||||
|
<string name="omnipod_error_suspend_delivery_failed_uncertain">Suspending delivery might have failed. Please manually refresh the Pod status from the Omnipod tab.</string>
|
||||||
|
<string name="omnipod_error_suspend_delivery_failed">Suspending delivery failed.</string>
|
||||||
<string name="omnipod_error_set_temp_basal_failed_old_tbr_might_be_cancelled">Setting temp basal failed. If a temp basal was previously running, it might have been cancelled. Please manually refresh the Pod status from the Omnipod tab.</string>
|
<string name="omnipod_error_set_temp_basal_failed_old_tbr_might_be_cancelled">Setting temp basal failed. If a temp basal was previously running, it might have been cancelled. Please manually refresh the Pod status from the Omnipod tab.</string>
|
||||||
<string name="omnipod_error_set_temp_basal_failed_old_tbr_cancelled_new_might_have_failed">Setting temp might have basal failed. If a temp basal was previously running, it has been cancelled. Please manually refresh the Pod status from the Omnipod tab.</string>
|
<string name="omnipod_error_set_temp_basal_failed_old_tbr_cancelled_new_might_have_failed">Setting temp might have basal failed. If a temp basal was previously running, it has been cancelled. Please manually refresh the Pod status from the Omnipod tab.</string>
|
||||||
<string name="omnipod_error_set_time_failed_delivery_might_be_suspended">Setting time might have failed. Delivery might be suspended! Please manually refresh the Pod status from the Omnipod tab and resume delivery if needed.</string>
|
<string name="omnipod_error_set_time_failed_delivery_might_be_suspended">Setting time might have failed. Delivery might be suspended! Please manually refresh the Pod status from the Omnipod tab and resume delivery if needed.</string>
|
||||||
|
@ -197,6 +199,21 @@
|
||||||
<string name="omnipod_error_set_initial_basal_schedule_no_profile">No basal profile is active. Make sure to activate your basal profile.</string>
|
<string name="omnipod_error_set_initial_basal_schedule_no_profile">No basal profile is active. Make sure to activate your basal profile.</string>
|
||||||
<string name="omnipod_pod_time_on_pod">Time on Pod</string>
|
<string name="omnipod_pod_time_on_pod">Time on Pod</string>
|
||||||
<string name="omnipod_set_time">Set time</string>
|
<string name="omnipod_set_time">Set time</string>
|
||||||
|
<string name="omnipod_unknown_custom_command">Unknown custom command: %1$s</string>
|
||||||
|
<string name="omnipod_failed_to_retrieve_pulse_log">Failed to retrieve Pulse Log</string>
|
||||||
|
<string name="omnipod_failed_to_refresh_status">Failed to refresh status</string>
|
||||||
|
<string name="omnipod_failed_to_acknowledge_alerts">Failed to acknowledge alerts</string>
|
||||||
|
<string name="omnipod_failed_to_suspend_delivery">Failed to suspend delivery</string>
|
||||||
|
<string name="omnipod_failed_to_set_time">Failed to set time</string>
|
||||||
|
<string name="omnipod_failed_to_resume_delivery">Failed to resume delivery</string>
|
||||||
|
<string name="omnipod_pulse_log">Pulse log</string>
|
||||||
|
<string name="omnipod_pulse_log_value">Pulse Log (copied to clipboard)</string>
|
||||||
|
<string name="omnipod_two_strings_concatenated_by_colon" translatable="false">%1$s: %2$s</string>
|
||||||
|
<string name="omnipod_confirmation">Confirmation</string>
|
||||||
|
<string name="omnipod_time_on_pod_updated">The time on Pod updated.</string>
|
||||||
|
<string name="omnipod_suspended_delivery">All insulin delivery suspended.</string>
|
||||||
|
<string name="omnipod_acknowledged_alerts">Acknowledged alerts.</string>
|
||||||
|
<string name="omnipod_delivery_resumed">Insulin delivery resumed.</string>
|
||||||
|
|
||||||
<plurals name="omnipod_minutes">
|
<plurals name="omnipod_minutes">
|
||||||
<item quantity="one">%1$d minute</item>
|
<item quantity="one">%1$d minute</item>
|
||||||
|
|
Loading…
Reference in a new issue