Merge pull request #2736 from jbr7rr/medtrum-300U
Medtrum: Support for 300U and fixes
This commit is contained in:
commit
d5c7489ecb
19 changed files with 364 additions and 242 deletions
|
@ -410,10 +410,33 @@ enum class PumpType {
|
||||||
baseBasalSpecialSteps = null,
|
baseBasalSpecialSteps = null,
|
||||||
pumpCapability = PumpCapability.MedtrumCapabilities,
|
pumpCapability = PumpCapability.MedtrumCapabilities,
|
||||||
isPatchPump = true,
|
isPatchPump = true,
|
||||||
maxReservoirReading = 400,
|
maxReservoirReading = 200,
|
||||||
source = Source.Medtrum
|
source = Source.Medtrum
|
||||||
),
|
),
|
||||||
MEDTRUM_UNTESTED(description = "Medtrum untested", model = "untested", parent = MEDTRUM_NANO);
|
MEDTRUM_300U(
|
||||||
|
description = "Medtrum 300U",
|
||||||
|
manufacturer = ManufacturerType.Medtrum,
|
||||||
|
model = "300U",
|
||||||
|
bolusSize = 0.05,
|
||||||
|
specialBolusSize = null,
|
||||||
|
extendedBolusSettings = DoseSettings(0.05, 30, 8 * 60, 0.05, 30.0),
|
||||||
|
pumpTempBasalType = PumpTempBasalType.Absolute,
|
||||||
|
tbrSettings = DoseSettings(0.05, 30, 12 * 60, 0.0, 30.0),
|
||||||
|
specialBasalDurations = PumpCapability.BasalRate_Duration30minAllowed,
|
||||||
|
baseBasalMinValue = 0.05,
|
||||||
|
baseBasalMaxValue = 30.0,
|
||||||
|
baseBasalStep = 0.05,
|
||||||
|
baseBasalSpecialSteps = null,
|
||||||
|
pumpCapability = PumpCapability.MedtrumCapabilities,
|
||||||
|
isPatchPump = true,
|
||||||
|
maxReservoirReading = 300,
|
||||||
|
source = Source.Medtrum
|
||||||
|
),
|
||||||
|
MEDTRUM_UNTESTED(
|
||||||
|
description = "Medtrum untested",
|
||||||
|
model = "untested",
|
||||||
|
parent = MEDTRUM_NANO
|
||||||
|
);
|
||||||
|
|
||||||
val description: String
|
val description: String
|
||||||
var manufacturer: ManufacturerType? = null
|
var manufacturer: ManufacturerType? = null
|
||||||
|
@ -455,8 +478,10 @@ enum class PumpType {
|
||||||
var hasCustomUnreachableAlertCheck = false
|
var hasCustomUnreachableAlertCheck = false
|
||||||
private set
|
private set
|
||||||
var isPatchPump = false
|
var isPatchPump = false
|
||||||
|
get() = parent?.isPatchPump ?: field
|
||||||
private set
|
private set
|
||||||
var maxReservoirReading = 50
|
var maxReservoirReading = 50
|
||||||
|
get() = parent?.maxReservoirReading ?: field
|
||||||
private set
|
private set
|
||||||
var supportBatteryLevel = true
|
var supportBatteryLevel = true
|
||||||
private set
|
private set
|
||||||
|
|
|
@ -60,6 +60,7 @@ fun PumpType.Companion.fromDbPumpType(pt: InterfaceIDs.PumpType): PumpType =
|
||||||
InterfaceIDs.PumpType.DIACONN_G8 -> PumpType.DIACONN_G8
|
InterfaceIDs.PumpType.DIACONN_G8 -> PumpType.DIACONN_G8
|
||||||
InterfaceIDs.PumpType.EOPATCH2 -> PumpType.EOFLOW_EOPATCH2
|
InterfaceIDs.PumpType.EOPATCH2 -> PumpType.EOFLOW_EOPATCH2
|
||||||
InterfaceIDs.PumpType.MEDTRUM -> PumpType.MEDTRUM_NANO
|
InterfaceIDs.PumpType.MEDTRUM -> PumpType.MEDTRUM_NANO
|
||||||
|
InterfaceIDs.PumpType.MEDTRUM_300U -> PumpType.MEDTRUM_300U
|
||||||
InterfaceIDs.PumpType.MEDTRUM_UNTESTED -> PumpType.MEDTRUM_UNTESTED
|
InterfaceIDs.PumpType.MEDTRUM_UNTESTED -> PumpType.MEDTRUM_UNTESTED
|
||||||
InterfaceIDs.PumpType.CACHE -> PumpType.CACHE
|
InterfaceIDs.PumpType.CACHE -> PumpType.CACHE
|
||||||
}
|
}
|
||||||
|
@ -80,6 +81,7 @@ fun PumpType.Source.toDbSource(): UserEntry.Sources =
|
||||||
PumpType.Source.OmnipodEros -> UserEntry.Sources.OmnipodEros
|
PumpType.Source.OmnipodEros -> UserEntry.Sources.OmnipodEros
|
||||||
PumpType.Source.OmnipodDash -> UserEntry.Sources.OmnipodDash
|
PumpType.Source.OmnipodDash -> UserEntry.Sources.OmnipodDash
|
||||||
PumpType.Source.EOPatch2 -> UserEntry.Sources.EOPatch2
|
PumpType.Source.EOPatch2 -> UserEntry.Sources.EOPatch2
|
||||||
|
PumpType.Source.Medtrum -> UserEntry.Sources.Medtrum
|
||||||
PumpType.Source.MDI -> UserEntry.Sources.MDI
|
PumpType.Source.MDI -> UserEntry.Sources.MDI
|
||||||
PumpType.Source.VirtualPump -> UserEntry.Sources.VirtualPump
|
PumpType.Source.VirtualPump -> UserEntry.Sources.VirtualPump
|
||||||
else -> UserEntry.Sources.Unknown
|
else -> UserEntry.Sources.Unknown
|
||||||
|
@ -120,6 +122,7 @@ fun PumpType.toDbPumpType(): InterfaceIDs.PumpType =
|
||||||
PumpType.DIACONN_G8 -> InterfaceIDs.PumpType.DIACONN_G8
|
PumpType.DIACONN_G8 -> InterfaceIDs.PumpType.DIACONN_G8
|
||||||
PumpType.EOFLOW_EOPATCH2 -> InterfaceIDs.PumpType.EOPATCH2
|
PumpType.EOFLOW_EOPATCH2 -> InterfaceIDs.PumpType.EOPATCH2
|
||||||
PumpType.MEDTRUM_NANO -> InterfaceIDs.PumpType.MEDTRUM
|
PumpType.MEDTRUM_NANO -> InterfaceIDs.PumpType.MEDTRUM
|
||||||
|
PumpType.MEDTRUM_300U -> InterfaceIDs.PumpType.MEDTRUM_300U
|
||||||
PumpType.MEDTRUM_UNTESTED -> InterfaceIDs.PumpType.MEDTRUM_UNTESTED
|
PumpType.MEDTRUM_UNTESTED -> InterfaceIDs.PumpType.MEDTRUM_UNTESTED
|
||||||
PumpType.CACHE -> InterfaceIDs.PumpType.CACHE
|
PumpType.CACHE -> InterfaceIDs.PumpType.CACHE
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,14 @@ class ValidatingEditTextPreference(ctx: Context, attrs: AttributeSet, defStyleAt
|
||||||
holder.isDividerAllowedBelow = false
|
holder.isDividerAllowedBelow = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun setMinNumber(min: Int) {
|
||||||
|
this.validatorParameters.minNumber = min
|
||||||
|
}
|
||||||
|
|
||||||
|
fun setMaxNumber(max: Int) {
|
||||||
|
this.validatorParameters.maxNumber = max
|
||||||
|
}
|
||||||
|
|
||||||
private fun obtainValidatorParameters(attrs: AttributeSet): DefaultEditTextValidator.Parameters {
|
private fun obtainValidatorParameters(attrs: AttributeSet): DefaultEditTextValidator.Parameters {
|
||||||
val typedArray = context.obtainStyledAttributes(attrs, R.styleable.FormEditText, 0, 0)
|
val typedArray = context.obtainStyledAttributes(attrs, R.styleable.FormEditText, 0, 0)
|
||||||
return DefaultEditTextValidator.Parameters(
|
return DefaultEditTextValidator.Parameters(
|
||||||
|
|
|
@ -172,6 +172,7 @@ data class UserEntry(
|
||||||
OmnipodEros,
|
OmnipodEros,
|
||||||
OmnipodDash, //No entry currently
|
OmnipodDash, //No entry currently
|
||||||
EOPatch2,
|
EOPatch2,
|
||||||
|
Medtrum,
|
||||||
MDI,
|
MDI,
|
||||||
VirtualPump,
|
VirtualPump,
|
||||||
SMS, //From SMS plugin
|
SMS, //From SMS plugin
|
||||||
|
|
|
@ -44,6 +44,7 @@ data class InterfaceIDs(
|
||||||
DIACONN_G8,
|
DIACONN_G8,
|
||||||
EOPATCH2,
|
EOPATCH2,
|
||||||
MEDTRUM,
|
MEDTRUM,
|
||||||
|
MEDTRUM_300U,
|
||||||
MEDTRUM_UNTESTED,
|
MEDTRUM_UNTESTED,
|
||||||
USER,
|
USER,
|
||||||
CACHE;
|
CACHE;
|
||||||
|
|
|
@ -97,6 +97,7 @@ class UserEntryPresentationHelperImpl @Inject constructor(
|
||||||
Sources.OmnipodEros -> R.drawable.ic_patch_pump_outline
|
Sources.OmnipodEros -> R.drawable.ic_patch_pump_outline
|
||||||
Sources.OmnipodDash -> R.drawable.ic_patch_pump_outline
|
Sources.OmnipodDash -> R.drawable.ic_patch_pump_outline
|
||||||
Sources.EOPatch2 -> info.nightscout.core.ui.R.drawable.ic_eopatch2_128
|
Sources.EOPatch2 -> info.nightscout.core.ui.R.drawable.ic_eopatch2_128
|
||||||
|
Sources.Medtrum -> info.nightscout.core.ui.R.drawable.ic_medtrum_128
|
||||||
Sources.MDI -> R.drawable.ic_ict
|
Sources.MDI -> R.drawable.ic_ict
|
||||||
Sources.VirtualPump -> R.drawable.ic_virtual_pump
|
Sources.VirtualPump -> R.drawable.ic_virtual_pump
|
||||||
Sources.SMS -> R.drawable.ic_sms
|
Sources.SMS -> R.drawable.ic_sms
|
||||||
|
|
|
@ -15,6 +15,7 @@ import dagger.android.HasAndroidInjector
|
||||||
import info.nightscout.core.ui.dialogs.OKDialog
|
import info.nightscout.core.ui.dialogs.OKDialog
|
||||||
import info.nightscout.core.ui.toast.ToastUtils
|
import info.nightscout.core.ui.toast.ToastUtils
|
||||||
import info.nightscout.core.utils.fabric.FabricPrivacy
|
import info.nightscout.core.utils.fabric.FabricPrivacy
|
||||||
|
import info.nightscout.core.validators.ValidatingEditTextPreference
|
||||||
import info.nightscout.interfaces.constraints.Constraint
|
import info.nightscout.interfaces.constraints.Constraint
|
||||||
import info.nightscout.interfaces.constraints.Constraints
|
import info.nightscout.interfaces.constraints.Constraints
|
||||||
import info.nightscout.interfaces.notifications.Notification
|
import info.nightscout.interfaces.notifications.Notification
|
||||||
|
@ -127,17 +128,25 @@ import kotlin.math.abs
|
||||||
|
|
||||||
override fun preprocessPreferences(preferenceFragment: PreferenceFragmentCompat) {
|
override fun preprocessPreferences(preferenceFragment: PreferenceFragmentCompat) {
|
||||||
super.preprocessPreferences(preferenceFragment)
|
super.preprocessPreferences(preferenceFragment)
|
||||||
|
|
||||||
|
preprocessSerialSettings(preferenceFragment)
|
||||||
|
preprocessAlarmSettings(preferenceFragment)
|
||||||
|
preprocessMaxInsulinSettings(preferenceFragment)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun preprocessSerialSettings(preferenceFragment: PreferenceFragmentCompat) {
|
||||||
val serialSetting = preferenceFragment.findPreference<EditTextPreference>(rh.gs(R.string.key_sn_input))
|
val serialSetting = preferenceFragment.findPreference<EditTextPreference>(rh.gs(R.string.key_sn_input))
|
||||||
serialSetting?.isEnabled = !isInitialized()
|
serialSetting?.apply {
|
||||||
serialSetting?.setOnBindEditTextListener { editText ->
|
isEnabled = !isInitialized()
|
||||||
|
setOnBindEditTextListener { editText ->
|
||||||
editText.addTextChangedListener(object : TextWatcher {
|
editText.addTextChangedListener(object : TextWatcher {
|
||||||
override fun afterTextChanged(newValue: Editable?) {
|
override fun afterTextChanged(newValue: Editable?) {
|
||||||
val newSN = newValue.toString().toLongOrNull(radix = 16)
|
val newSN = newValue?.toString()?.toLongOrNull(radix = 16) ?: 0
|
||||||
val newDeviceType = MedtrumSnUtil().getDeviceTypeFromSerial(newSN ?: 0)
|
val newDeviceType = MedtrumSnUtil().getDeviceTypeFromSerial(newSN)
|
||||||
if (newDeviceType == MedtrumSnUtil.INVALID) {
|
editText.error = if (newDeviceType == MedtrumSnUtil.INVALID) {
|
||||||
editText.error = rh.gs(R.string.sn_input_invalid)
|
rh.gs(R.string.sn_input_invalid)
|
||||||
} else {
|
} else {
|
||||||
editText.error = null
|
null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -150,10 +159,10 @@ import kotlin.math.abs
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
serialSetting?.setOnPreferenceChangeListener { _, newValue ->
|
setOnPreferenceChangeListener { _, newValue ->
|
||||||
if (newValue is String) {
|
val newSN = (newValue as? String)?.toLongOrNull(radix = 16) ?: 0
|
||||||
val newSN = newValue.toLongOrNull(radix = 16)
|
val newDeviceType = MedtrumSnUtil().getDeviceTypeFromSerial(newSN)
|
||||||
val newDeviceType = MedtrumSnUtil().getDeviceTypeFromSerial(newSN ?: 0)
|
|
||||||
when {
|
when {
|
||||||
newDeviceType == MedtrumSnUtil.INVALID -> {
|
newDeviceType == MedtrumSnUtil.INVALID -> {
|
||||||
preferenceFragment.activity?.let { activity ->
|
preferenceFragment.activity?.let { activity ->
|
||||||
|
@ -171,30 +180,61 @@ import kotlin.math.abs
|
||||||
|
|
||||||
else -> true
|
else -> true
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
false
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun preprocessAlarmSettings(preferenceFragment: PreferenceFragmentCompat) {
|
||||||
val alarmSetting = preferenceFragment.findPreference<ListPreference>(rh.gs(R.string.key_alarm_setting))
|
val alarmSetting = preferenceFragment.findPreference<ListPreference>(rh.gs(R.string.key_alarm_setting))
|
||||||
val allAlarmEntries = preferenceFragment.resources.getStringArray(R.array.alarmSettings)
|
val allAlarmEntries = preferenceFragment.resources.getStringArray(R.array.alarmSettings)
|
||||||
val allAlarmValues = preferenceFragment.resources.getStringArray(R.array.alarmSettingsValues)
|
val allAlarmValues = preferenceFragment.resources.getStringArray(R.array.alarmSettingsValues)
|
||||||
|
|
||||||
if (allAlarmEntries.size < 8 || allAlarmValues.size < 8) {
|
if (allAlarmEntries.size < 8 || allAlarmValues.size < 8) {
|
||||||
aapsLogger.error(LTag.PUMP, "Alarm settings array is not complete")
|
aapsLogger.error(LTag.PUMP, "Alarm settings array is not complete")
|
||||||
return
|
} else {
|
||||||
}
|
|
||||||
|
|
||||||
when (medtrumPump.pumpType()) {
|
when (medtrumPump.pumpType()) {
|
||||||
PumpType.MEDTRUM_NANO -> {
|
PumpType.MEDTRUM_NANO, PumpType.MEDTRUM_300U -> {
|
||||||
alarmSetting?.entries = arrayOf(allAlarmEntries[6], allAlarmEntries[7]) // "Beep", "Silent"
|
alarmSetting?.apply {
|
||||||
alarmSetting?.entryValues = arrayOf(allAlarmValues[6], allAlarmValues[7]) // "6", "7"
|
entries = arrayOf(allAlarmEntries[6], allAlarmEntries[7]) // "Beep", "Silent"
|
||||||
|
entryValues = arrayOf(allAlarmValues[6], allAlarmValues[7]) // "6", "7"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
// Use Nano settings for unknown pumps
|
// Use default
|
||||||
alarmSetting?.entries = arrayOf(allAlarmEntries[6], allAlarmEntries[7]) // "Beep", "Silent"
|
}
|
||||||
alarmSetting?.entryValues = arrayOf(allAlarmValues[6], allAlarmValues[7]) // "6", "7"
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun preprocessMaxInsulinSettings(preferenceFragment: PreferenceFragmentCompat) {
|
||||||
|
val hourlyMaxSetting = preferenceFragment.findPreference<ValidatingEditTextPreference>(rh.gs(R.string.key_hourly_max_insulin))
|
||||||
|
val dailyMaxSetting = preferenceFragment.findPreference<ValidatingEditTextPreference>(rh.gs(R.string.key_daily_max_insulin))
|
||||||
|
|
||||||
|
val hourlyMaxValue = hourlyMaxSetting?.text?.toIntOrNull() ?: 0
|
||||||
|
val newDailyMaxMinValue = if (hourlyMaxValue > 20) hourlyMaxValue else 20
|
||||||
|
dailyMaxSetting?.setMinNumber(newDailyMaxMinValue)
|
||||||
|
|
||||||
|
val pumpTypeSettings = when (medtrumPump.pumpType()) {
|
||||||
|
PumpType.MEDTRUM_NANO -> Pair(40, 180) // maxHourlyMax, maxDailyMax
|
||||||
|
PumpType.MEDTRUM_300U -> Pair(60, 270)
|
||||||
|
else -> Pair(40, 180)
|
||||||
|
}
|
||||||
|
|
||||||
|
hourlyMaxSetting?.apply {
|
||||||
|
setMaxNumber(pumpTypeSettings.first)
|
||||||
|
val hourlyCurrentValue = text?.toIntOrNull() ?: 0
|
||||||
|
if (hourlyCurrentValue > pumpTypeSettings.first) {
|
||||||
|
text = pumpTypeSettings.first.toString()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dailyMaxSetting?.apply {
|
||||||
|
setMaxNumber(pumpTypeSettings.second)
|
||||||
|
val dailyCurrentValue = text?.toIntOrNull() ?: 0
|
||||||
|
when {
|
||||||
|
dailyCurrentValue < newDailyMaxMinValue -> text = newDailyMaxMinValue.toString()
|
||||||
|
dailyCurrentValue > pumpTypeSettings.second -> text = pumpTypeSettings.second.toString()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -220,6 +260,7 @@ import kotlin.math.abs
|
||||||
override fun isHandshakeInProgress(): Boolean = false
|
override fun isHandshakeInProgress(): Boolean = false
|
||||||
|
|
||||||
override fun finishHandshaking() {
|
override fun finishHandshaking() {
|
||||||
|
// Unused
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun connect(reason: String) {
|
override fun connect(reason: String) {
|
||||||
|
@ -417,6 +458,7 @@ import kotlin.math.abs
|
||||||
try {
|
try {
|
||||||
extended.put("ActiveProfile", profileName)
|
extended.put("ActiveProfile", profileName)
|
||||||
} catch (ignored: Exception) {
|
} catch (ignored: Exception) {
|
||||||
|
// Ignore
|
||||||
}
|
}
|
||||||
pumpJson.put("status", status)
|
pumpJson.put("status", status)
|
||||||
pumpJson.put("extended", extended)
|
pumpJson.put("extended", extended)
|
||||||
|
@ -472,6 +514,7 @@ import kotlin.math.abs
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun executeCustomAction(customActionType: CustomActionType) {
|
override fun executeCustomAction(customActionType: CustomActionType) {
|
||||||
|
// Unused
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun executeCustomCommand(customCommand: CustomCommand): PumpEnactResult? {
|
override fun executeCustomCommand(customCommand: CustomCommand): PumpEnactResult? {
|
||||||
|
|
|
@ -2,6 +2,7 @@ package info.nightscout.pump.medtrum
|
||||||
|
|
||||||
import android.util.Base64
|
import android.util.Base64
|
||||||
import info.nightscout.interfaces.profile.Profile
|
import info.nightscout.interfaces.profile.Profile
|
||||||
|
import info.nightscout.interfaces.pump.DetailedBolusInfo
|
||||||
import info.nightscout.interfaces.pump.PumpSync
|
import info.nightscout.interfaces.pump.PumpSync
|
||||||
import info.nightscout.interfaces.pump.TemporaryBasalStorage
|
import info.nightscout.interfaces.pump.TemporaryBasalStorage
|
||||||
import info.nightscout.interfaces.pump.defs.PumpType
|
import info.nightscout.interfaces.pump.defs.PumpType
|
||||||
|
@ -294,6 +295,7 @@ class MedtrumPump @Inject constructor(
|
||||||
fun pumpType(type: Int): PumpType =
|
fun pumpType(type: Int): PumpType =
|
||||||
when (type) {
|
when (type) {
|
||||||
MedtrumSnUtil.MD_0201, MedtrumSnUtil.MD_8201 -> PumpType.MEDTRUM_NANO
|
MedtrumSnUtil.MD_0201, MedtrumSnUtil.MD_8201 -> PumpType.MEDTRUM_NANO
|
||||||
|
MedtrumSnUtil.MD_8301 -> PumpType.MEDTRUM_300U
|
||||||
else -> PumpType.MEDTRUM_UNTESTED
|
else -> PumpType.MEDTRUM_UNTESTED
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -333,7 +335,7 @@ class MedtrumPump @Inject constructor(
|
||||||
|
|
||||||
val date = GregorianCalendar()
|
val date = GregorianCalendar()
|
||||||
date.timeInMillis = timestamp
|
date.timeInMillis = timestamp
|
||||||
val hourOfDayMinutes = date.get(GregorianCalendar.HOUR_OF_DAY) * 60 + date.get(GregorianCalendar.MINUTE)
|
val hourOfDayMinutes = date[GregorianCalendar.HOUR_OF_DAY] * 60 + date[GregorianCalendar.MINUTE]
|
||||||
|
|
||||||
for (index in 0 until basalCount) {
|
for (index in 0 until basalCount) {
|
||||||
val currentIndex = 1 + (index * 3)
|
val currentIndex = 1 + (index * 3)
|
||||||
|
@ -354,7 +356,6 @@ class MedtrumPump @Inject constructor(
|
||||||
aapsLogger.debug(LTag.PUMP, "getHourlyBasalFromMedtrumProfileArray: basal: $basal")
|
aapsLogger.debug(LTag.PUMP, "getHourlyBasalFromMedtrumProfileArray: basal: $basal")
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
// aapsLogger.debug(LTag.PUMP, "getHourlyBasalFromMedtrumProfileArray: rate: $rate, startMinutes: $startMinutes, endMinutes: $endMinutes")
|
|
||||||
}
|
}
|
||||||
return basal
|
return basal
|
||||||
}
|
}
|
||||||
|
@ -378,7 +379,8 @@ class MedtrumPump @Inject constructor(
|
||||||
)
|
)
|
||||||
@Suppress("UNNECESSARY_SAFE_CALL") // Safe call to allow mocks to return null
|
@Suppress("UNNECESSARY_SAFE_CALL") // Safe call to allow mocks to return null
|
||||||
val expectedTemporaryBasal = pumpSync.expectedPumpState()?.temporaryBasal
|
val expectedTemporaryBasal = pumpSync.expectedPumpState()?.temporaryBasal
|
||||||
if (basalType.isTempBasal() && expectedTemporaryBasal?.pumpId != basalStartTime) {
|
when {
|
||||||
|
basalType.isTempBasal() && expectedTemporaryBasal?.pumpId != basalStartTime -> {
|
||||||
// Note: temporaryBasalInfo will be removed from temporaryBasalStorage after this call
|
// Note: temporaryBasalInfo will be removed from temporaryBasalStorage after this call
|
||||||
val temporaryBasalInfo = temporaryBasalStorage.findTemporaryBasal(basalStartTime, basalRate)
|
val temporaryBasalInfo = temporaryBasalStorage.findTemporaryBasal(basalStartTime, basalRate)
|
||||||
|
|
||||||
|
@ -403,7 +405,9 @@ class MedtrumPump @Inject constructor(
|
||||||
LTag.PUMPCOMM,
|
LTag.PUMPCOMM,
|
||||||
"handleBasalStatusUpdate: ${if (newRecord) "**NEW** " else ""}EVENT TEMP_START ($basalType) ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) " + "Rate: $basalRate Duration: ${duration} temporaryBasalInfo: $temporaryBasalInfo, expectedTemporaryBasal: $expectedTemporaryBasal"
|
"handleBasalStatusUpdate: ${if (newRecord) "**NEW** " else ""}EVENT TEMP_START ($basalType) ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) " + "Rate: $basalRate Duration: ${duration} temporaryBasalInfo: $temporaryBasalInfo, expectedTemporaryBasal: $expectedTemporaryBasal"
|
||||||
)
|
)
|
||||||
} else if (basalType.isSuspendedByPump() && expectedTemporaryBasal?.pumpId != basalStartTime) {
|
}
|
||||||
|
|
||||||
|
basalType.isSuspendedByPump() && expectedTemporaryBasal?.pumpId != basalStartTime -> {
|
||||||
val newRecord = pumpSync.syncTemporaryBasalWithPumpId(
|
val newRecord = pumpSync.syncTemporaryBasalWithPumpId(
|
||||||
timestamp = basalStartTime,
|
timestamp = basalStartTime,
|
||||||
rate = 0.0,
|
rate = 0.0,
|
||||||
|
@ -418,10 +422,14 @@ class MedtrumPump @Inject constructor(
|
||||||
LTag.PUMPCOMM,
|
LTag.PUMPCOMM,
|
||||||
"handleBasalStatusUpdate: ${if (newRecord) "**NEW** " else ""}EVENT TEMP_START ($basalType) ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) expectedTemporaryBasal: $expectedTemporaryBasal"
|
"handleBasalStatusUpdate: ${if (newRecord) "**NEW** " else ""}EVENT TEMP_START ($basalType) ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) expectedTemporaryBasal: $expectedTemporaryBasal"
|
||||||
)
|
)
|
||||||
} else if (basalType == BasalType.NONE && expectedTemporaryBasal?.rate != basalRate && expectedTemporaryBasal?.duration != T.mins(FAKE_TBR_LENGTH).msecs()) {
|
}
|
||||||
|
|
||||||
|
basalType == BasalType.NONE && expectedTemporaryBasal?.rate != basalRate && expectedTemporaryBasal?.duration != T.mins(FAKE_TBR_LENGTH).msecs() -> {
|
||||||
// Pump suspended, set fake TBR
|
// Pump suspended, set fake TBR
|
||||||
setFakeTBR()
|
setFakeTBR()
|
||||||
} else if (basalType == BasalType.STANDARD) {
|
}
|
||||||
|
|
||||||
|
basalType == BasalType.STANDARD -> {
|
||||||
if (expectedTemporaryBasal != null) {
|
if (expectedTemporaryBasal != null) {
|
||||||
// Pump resumed, sync end
|
// Pump resumed, sync end
|
||||||
val success = pumpSync.syncStopTemporaryBasalWithPumpId(
|
val success = pumpSync.syncStopTemporaryBasalWithPumpId(
|
||||||
|
@ -433,6 +441,7 @@ class MedtrumPump @Inject constructor(
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "handleBasalStatusUpdate: EVENT TEMP_END ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) success: $success")
|
aapsLogger.debug(LTag.PUMPCOMM, "handleBasalStatusUpdate: EVENT TEMP_END ${dateUtil.dateAndTimeString(basalStartTime)} ($basalStartTime) success: $success")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Update medtrum pump state
|
// Update medtrum pump state
|
||||||
_lastBasalType.value = basalType
|
_lastBasalType.value = basalType
|
||||||
|
@ -529,6 +538,26 @@ class MedtrumPump @Inject constructor(
|
||||||
return rh.gs(stringId)
|
return rh.gs(stringId)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun handleNewPatch(newPatchId: Long, sequenceNumber: Int, newStartTime: Long) {
|
||||||
|
patchId = newPatchId
|
||||||
|
patchStartTime = newStartTime
|
||||||
|
currentSequenceNumber = sequenceNumber // We are activated, set the new seq nr
|
||||||
|
syncedSequenceNumber = 1 // Always start with 1
|
||||||
|
// Sync cannula change
|
||||||
|
pumpSync.insertTherapyEventIfNewWithTimestamp(
|
||||||
|
timestamp = newStartTime,
|
||||||
|
type = DetailedBolusInfo.EventType.CANNULA_CHANGE,
|
||||||
|
pumpType = pumpType(),
|
||||||
|
pumpSerial = pumpSN.toString(radix = 16)
|
||||||
|
)
|
||||||
|
pumpSync.insertTherapyEventIfNewWithTimestamp(
|
||||||
|
timestamp = newStartTime,
|
||||||
|
type = DetailedBolusInfo.EventType.INSULIN_CHANGE,
|
||||||
|
pumpType = pumpType(),
|
||||||
|
pumpSerial = pumpSN.toString(radix = 16)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
private fun saveActiveAlarms() {
|
private fun saveActiveAlarms() {
|
||||||
val alarmsStr = activeAlarms.joinToString(separator = ",") { it.name }
|
val alarmsStr = activeAlarms.joinToString(separator = ",") { it.name }
|
||||||
sp.putString(R.string.key_active_alarms, alarmsStr)
|
sp.putString(R.string.key_active_alarms, alarmsStr)
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package info.nightscout.pump.medtrum.comm.packets
|
package info.nightscout.pump.medtrum.comm.packets
|
||||||
|
|
||||||
import dagger.android.HasAndroidInjector
|
import dagger.android.HasAndroidInjector
|
||||||
import info.nightscout.interfaces.pump.DetailedBolusInfo
|
|
||||||
import info.nightscout.interfaces.pump.PumpSync
|
import info.nightscout.interfaces.pump.PumpSync
|
||||||
import info.nightscout.pump.medtrum.MedtrumPump
|
import info.nightscout.pump.medtrum.MedtrumPump
|
||||||
import info.nightscout.pump.medtrum.comm.enums.CommandType.ACTIVATE
|
import info.nightscout.pump.medtrum.comm.enums.CommandType.ACTIVATE
|
||||||
|
@ -93,24 +92,8 @@ class ActivatePacket(injector: HasAndroidInjector, private val basalProfile: Byt
|
||||||
val basalPatchId = data.copyOfRange(RESP_BASAL_PATCH_ID_START, RESP_BASAL_PATCH_ID_END).toLong()
|
val basalPatchId = data.copyOfRange(RESP_BASAL_PATCH_ID_START, RESP_BASAL_PATCH_ID_END).toLong()
|
||||||
val basalStartTime = medtrumTimeUtil.convertPumpTimeToSystemTimeMillis(data.copyOfRange(RESP_BASAL_START_TIME_START, RESP_BASAL_START_TIME_END).toLong())
|
val basalStartTime = medtrumTimeUtil.convertPumpTimeToSystemTimeMillis(data.copyOfRange(RESP_BASAL_START_TIME_START, RESP_BASAL_START_TIME_END).toLong())
|
||||||
|
|
||||||
medtrumPump.patchId = patchId
|
|
||||||
medtrumPump.lastTimeReceivedFromPump = time
|
medtrumPump.lastTimeReceivedFromPump = time
|
||||||
medtrumPump.currentSequenceNumber = basalSequence // We are activated, set the new seq nr
|
medtrumPump.handleNewPatch(patchId, basalSequence, System.currentTimeMillis())
|
||||||
medtrumPump.syncedSequenceNumber = basalSequence // We are activated, reset the synced seq nr ()
|
|
||||||
|
|
||||||
// Sync cannula change
|
|
||||||
pumpSync.insertTherapyEventIfNewWithTimestamp(
|
|
||||||
timestamp = System.currentTimeMillis(),
|
|
||||||
type = DetailedBolusInfo.EventType.CANNULA_CHANGE,
|
|
||||||
pumpType = medtrumPump.pumpType(),
|
|
||||||
pumpSerial = medtrumPump.pumpSN.toString(radix = 16)
|
|
||||||
)
|
|
||||||
pumpSync.insertTherapyEventIfNewWithTimestamp(
|
|
||||||
timestamp = System.currentTimeMillis(),
|
|
||||||
type = DetailedBolusInfo.EventType.INSULIN_CHANGE,
|
|
||||||
pumpType = medtrumPump.pumpType(),
|
|
||||||
pumpSerial = medtrumPump.pumpSN.toString(radix = 16)
|
|
||||||
)
|
|
||||||
|
|
||||||
// Update the actual basal profile
|
// Update the actual basal profile
|
||||||
medtrumPump.actualBasalProfile = basalProfile
|
medtrumPump.actualBasalProfile = basalProfile
|
||||||
|
|
|
@ -111,7 +111,8 @@ class GetRecordPacket(injector: HasAndroidInjector, private val recordIndex: Int
|
||||||
"$bolusExtendedDuration, " + "bolusExtendedDelivered: $bolusExtendedDelivered, bolusCarb: $bolusCarb, bolusGlucose: $bolusGlucose, bolusIOB: $bolusIOB, unknown1: $unknown1, unknown2: $unknown2, " + "bolusType: $bolusType, bolusWizard: $bolusWizard"
|
"$bolusExtendedDuration, " + "bolusExtendedDelivered: $bolusExtendedDelivered, bolusCarb: $bolusCarb, bolusGlucose: $bolusGlucose, bolusIOB: $bolusIOB, unknown1: $unknown1, unknown2: $unknown2, " + "bolusType: $bolusType, bolusWizard: $bolusWizard"
|
||||||
)
|
)
|
||||||
|
|
||||||
if (bolusType == BolusType.NORMAL) {
|
when (bolusType) {
|
||||||
|
BolusType.NORMAL -> {
|
||||||
val detailedBolusInfo = detailedBolusInfoStorage.findDetailedBolusInfo(bolusStartTime, bolusNormalDelivered)
|
val detailedBolusInfo = detailedBolusInfoStorage.findDetailedBolusInfo(bolusStartTime, bolusNormalDelivered)
|
||||||
var newRecord = false
|
var newRecord = false
|
||||||
if (detailedBolusInfo != null) {
|
if (detailedBolusInfo != null) {
|
||||||
|
@ -148,7 +149,9 @@ class GetRecordPacket(injector: HasAndroidInjector, private val recordIndex: Int
|
||||||
medtrumPump.lastBolusTime = bolusStartTime
|
medtrumPump.lastBolusTime = bolusStartTime
|
||||||
medtrumPump.lastBolusAmount = bolusNormalDelivered
|
medtrumPump.lastBolusAmount = bolusNormalDelivered
|
||||||
}
|
}
|
||||||
} else if (bolusType == BolusType.EXTENDED) {
|
}
|
||||||
|
|
||||||
|
BolusType.EXTENDED -> {
|
||||||
val newRecord = pumpSync.syncExtendedBolusWithPumpId(
|
val newRecord = pumpSync.syncExtendedBolusWithPumpId(
|
||||||
timestamp = bolusStartTime,
|
timestamp = bolusStartTime,
|
||||||
amount = bolusExtendedDelivered,
|
amount = bolusExtendedDelivered,
|
||||||
|
@ -162,7 +165,9 @@ class GetRecordPacket(injector: HasAndroidInjector, private val recordIndex: Int
|
||||||
LTag.PUMPCOMM,
|
LTag.PUMPCOMM,
|
||||||
"from record: ${if (newRecord) "**NEW** " else ""}EVENT EXTENDED BOLUS ${dateUtil.dateAndTimeString(bolusStartTime)} ($bolusStartTime) Bolus: ${bolusNormalDelivered}U "
|
"from record: ${if (newRecord) "**NEW** " else ""}EVENT EXTENDED BOLUS ${dateUtil.dateAndTimeString(bolusStartTime)} ($bolusStartTime) Bolus: ${bolusNormalDelivered}U "
|
||||||
)
|
)
|
||||||
} else if (bolusType == BolusType.COMBI) {
|
}
|
||||||
|
|
||||||
|
BolusType.COMBI -> {
|
||||||
// Note, this should never happen, as we don't use combo bolus
|
// Note, this should never happen, as we don't use combo bolus
|
||||||
val detailedBolusInfo = detailedBolusInfoStorage.findDetailedBolusInfo(bolusStartTime, bolusNormalDelivered)
|
val detailedBolusInfo = detailedBolusInfoStorage.findDetailedBolusInfo(bolusStartTime, bolusNormalDelivered)
|
||||||
val newRecord = pumpSync.syncBolusWithPumpId(
|
val newRecord = pumpSync.syncBolusWithPumpId(
|
||||||
|
@ -194,9 +199,12 @@ class GetRecordPacket(injector: HasAndroidInjector, private val recordIndex: Int
|
||||||
medtrumPump.lastBolusTime = bolusStartTime
|
medtrumPump.lastBolusTime = bolusStartTime
|
||||||
medtrumPump.lastBolusAmount = bolusNormalDelivered
|
medtrumPump.lastBolusAmount = bolusNormalDelivered
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
aapsLogger.error(LTag.PUMPCOMM, "GetRecordPacket HandleResponse: BOLUS_RECORD: Unknown bolus type: $bolusType")
|
aapsLogger.error(LTag.PUMPCOMM, "GetRecordPacket HandleResponse: BOLUS_RECORD: Unknown bolus type: $bolusType")
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -85,6 +85,7 @@ class NotificationPacket(val injector: HasAndroidInjector) {
|
||||||
fun handleMaskedMessage(data: ByteArray) {
|
fun handleMaskedMessage(data: ByteArray) {
|
||||||
val fieldMask = data.copyOfRange(0, 2).toInt()
|
val fieldMask = data.copyOfRange(0, 2).toInt()
|
||||||
var offset = 2
|
var offset = 2
|
||||||
|
var newPatchStartTime: Long? = null
|
||||||
|
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "Message field mask: $fieldMask")
|
aapsLogger.debug(LTag.PUMPCOMM, "Message field mask: $fieldMask")
|
||||||
|
|
||||||
|
@ -148,12 +149,12 @@ class NotificationPacket(val injector: HasAndroidInjector) {
|
||||||
|
|
||||||
if (fieldMask and MASK_START_TIME != 0) {
|
if (fieldMask and MASK_START_TIME != 0) {
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "Start time notification received")
|
aapsLogger.debug(LTag.PUMPCOMM, "Start time notification received")
|
||||||
val patchStartTime = MedtrumTimeUtil().convertPumpTimeToSystemTimeMillis(data.copyOfRange(offset, offset + 4).toLong())
|
newPatchStartTime = MedtrumTimeUtil().convertPumpTimeToSystemTimeMillis(data.copyOfRange(offset, offset + 4).toLong())
|
||||||
if (medtrumPump.patchStartTime != patchStartTime) {
|
if (medtrumPump.patchStartTime != newPatchStartTime) {
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "Patch start time changed from ${medtrumPump.patchStartTime} to $patchStartTime")
|
aapsLogger.debug(LTag.PUMPCOMM, "Patch start time changed from ${medtrumPump.patchStartTime} to $newPatchStartTime")
|
||||||
medtrumPump.patchStartTime = patchStartTime
|
medtrumPump.patchStartTime = newPatchStartTime
|
||||||
}
|
}
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "Patch start time: ${patchStartTime}")
|
aapsLogger.debug(LTag.PUMPCOMM, "Patch start time: $newPatchStartTime")
|
||||||
offset += 4
|
offset += 4
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,6 +177,11 @@ class NotificationPacket(val injector: HasAndroidInjector) {
|
||||||
val patchId = data.copyOfRange(offset + 2, offset + 4).toLong()
|
val patchId = data.copyOfRange(offset + 2, offset + 4).toLong()
|
||||||
if (patchId != medtrumPump.patchId) {
|
if (patchId != medtrumPump.patchId) {
|
||||||
aapsLogger.warn(LTag.PUMPCOMM, "handleMaskedMessage: We got wrong patch id!")
|
aapsLogger.warn(LTag.PUMPCOMM, "handleMaskedMessage: We got wrong patch id!")
|
||||||
|
if (newPatchStartTime != null) {
|
||||||
|
// This is a fallback for when the activate packet did not receive the ack but the patch activated anyway
|
||||||
|
aapsLogger.error(LTag.PUMPCOMM, "handleMaskedMessage: Also Received start time in this packet, registering new patch id: $patchId")
|
||||||
|
medtrumPump.handleNewPatch(patchId, sequence, newPatchStartTime)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "Last known sequence number: ${medtrumPump.currentSequenceNumber}, patch id: ${patchId}")
|
aapsLogger.debug(LTag.PUMPCOMM, "Last known sequence number: ${medtrumPump.currentSequenceNumber}, patch id: ${patchId}")
|
||||||
offset += 4
|
offset += 4
|
||||||
|
|
|
@ -39,11 +39,11 @@ class Crypt {
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun randomGen(input: Long): Long {
|
private fun randomGen(input: Long): Long {
|
||||||
val A = 16807
|
val a = 16807
|
||||||
val Q = 127773
|
val q = 127773
|
||||||
val R = 2836
|
val r = 2836
|
||||||
val tmp1 = input / Q
|
val tmp1 = input / q
|
||||||
var ret = (input - (tmp1 * Q)) * A - (tmp1 * R)
|
var ret = (input - (tmp1 * q)) * a - (tmp1 * r)
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
ret += 2147483647L
|
ret += 2147483647L
|
||||||
}
|
}
|
||||||
|
|
|
@ -310,9 +310,7 @@ class BLEComm @Inject internal constructor(
|
||||||
private fun readDescriptor(descriptor: BluetoothGattDescriptor?) {
|
private fun readDescriptor(descriptor: BluetoothGattDescriptor?) {
|
||||||
aapsLogger.debug(LTag.PUMPBTCOMM, "readDescriptor")
|
aapsLogger.debug(LTag.PUMPBTCOMM, "readDescriptor")
|
||||||
if (mBluetoothAdapter == null || mBluetoothGatt == null || descriptor == null) {
|
if (mBluetoothAdapter == null || mBluetoothGatt == null || descriptor == null) {
|
||||||
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
handleNotInitialized()
|
||||||
isConnecting = false
|
|
||||||
isConnected = false
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
mBluetoothGatt?.readDescriptor(descriptor)
|
mBluetoothGatt?.readDescriptor(descriptor)
|
||||||
|
@ -323,9 +321,7 @@ class BLEComm @Inject internal constructor(
|
||||||
aapsLogger.debug(LTag.PUMPBTCOMM, "checkDescriptor")
|
aapsLogger.debug(LTag.PUMPBTCOMM, "checkDescriptor")
|
||||||
val service = getGattService()
|
val service = getGattService()
|
||||||
if (mBluetoothAdapter == null || mBluetoothGatt == null || service == null) {
|
if (mBluetoothAdapter == null || mBluetoothGatt == null || service == null) {
|
||||||
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
handleNotInitialized()
|
||||||
isConnecting = false
|
|
||||||
isConnected = false
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (descriptor.value.toInt() > 0) {
|
if (descriptor.value.toInt() > 0) {
|
||||||
|
@ -352,9 +348,7 @@ class BLEComm @Inject internal constructor(
|
||||||
private fun setCharacteristicNotification(characteristic: BluetoothGattCharacteristic?, enabled: Boolean) {
|
private fun setCharacteristicNotification(characteristic: BluetoothGattCharacteristic?, enabled: Boolean) {
|
||||||
aapsLogger.debug(LTag.PUMPBTCOMM, "setCharacteristicNotification")
|
aapsLogger.debug(LTag.PUMPBTCOMM, "setCharacteristicNotification")
|
||||||
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
||||||
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
handleNotInitialized()
|
||||||
isConnecting = false
|
|
||||||
isConnected = false
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
mBluetoothGatt?.setCharacteristicNotification(characteristic, enabled)
|
mBluetoothGatt?.setCharacteristicNotification(characteristic, enabled)
|
||||||
|
@ -366,7 +360,7 @@ class BLEComm @Inject internal constructor(
|
||||||
it.value = BluetoothGattDescriptor.ENABLE_INDICATION_VALUE
|
it.value = BluetoothGattDescriptor.ENABLE_INDICATION_VALUE
|
||||||
mBluetoothGatt?.writeDescriptor(it)
|
mBluetoothGatt?.writeDescriptor(it)
|
||||||
} else {
|
} else {
|
||||||
|
// Do nothing
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -417,9 +411,7 @@ class BLEComm @Inject internal constructor(
|
||||||
private fun getGattService(): BluetoothGattService? {
|
private fun getGattService(): BluetoothGattService? {
|
||||||
aapsLogger.debug(LTag.PUMPBTCOMM, "getGattService")
|
aapsLogger.debug(LTag.PUMPBTCOMM, "getGattService")
|
||||||
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
||||||
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
handleNotInitialized()
|
||||||
isConnecting = false
|
|
||||||
isConnected = false
|
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
return mBluetoothGatt?.getService(UUID.fromString(SERVICE_UUID))
|
return mBluetoothGatt?.getService(UUID.fromString(SERVICE_UUID))
|
||||||
|
@ -431,9 +423,7 @@ class BLEComm @Inject internal constructor(
|
||||||
private fun writeCharacteristic(characteristic: BluetoothGattCharacteristic, data: ByteArray?) {
|
private fun writeCharacteristic(characteristic: BluetoothGattCharacteristic, data: ByteArray?) {
|
||||||
handler.postDelayed({
|
handler.postDelayed({
|
||||||
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
|
||||||
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
handleNotInitialized()
|
||||||
isConnecting = false
|
|
||||||
isConnected = false
|
|
||||||
} else {
|
} else {
|
||||||
characteristic.value = data
|
characteristic.value = data
|
||||||
characteristic.writeType = BluetoothGattCharacteristic.WRITE_TYPE_DEFAULT
|
characteristic.writeType = BluetoothGattCharacteristic.WRITE_TYPE_DEFAULT
|
||||||
|
@ -456,7 +446,7 @@ class BLEComm @Inject internal constructor(
|
||||||
var uuid: String
|
var uuid: String
|
||||||
val gattCharacteristics = gattService.characteristics
|
val gattCharacteristics = gattService.characteristics
|
||||||
for (i in 0 until gattCharacteristics.size) {
|
for (i in 0 until gattCharacteristics.size) {
|
||||||
val gattCharacteristic = gattCharacteristics.get(i)
|
val gattCharacteristic = gattCharacteristics[i]
|
||||||
// Check whether read or write properties is set, the pump needs us to enable notifications on all characteristics that have these properties
|
// Check whether read or write properties is set, the pump needs us to enable notifications on all characteristics that have these properties
|
||||||
if (gattCharacteristic.properties and NEEDS_ENABLE > 0) {
|
if (gattCharacteristic.properties and NEEDS_ENABLE > 0) {
|
||||||
handler.postDelayed({
|
handler.postDelayed({
|
||||||
|
@ -472,4 +462,11 @@ class BLEComm @Inject internal constructor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun handleNotInitialized() {
|
||||||
|
aapsLogger.error("BluetoothAdapter not initialized_ERROR")
|
||||||
|
isConnecting = false
|
||||||
|
isConnected = false
|
||||||
|
return
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -135,20 +135,13 @@ class MedtrumService : DaggerService(), BLECommCallback {
|
||||||
}
|
}
|
||||||
}, fabricPrivacy::logException)
|
}, fabricPrivacy::logException)
|
||||||
scope.launch {
|
scope.launch {
|
||||||
medtrumPump.pumpStateFlow.collect { state ->
|
medtrumPump.pumpStateFlow.collect { pumpState ->
|
||||||
handlePumpStateUpdate(state)
|
handlePumpStateUpdate(pumpState)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
scope.launch {
|
scope.launch {
|
||||||
medtrumPump.connectionStateFlow.collect { state ->
|
medtrumPump.connectionStateFlow.collect { connectionState ->
|
||||||
if (medtrumPlugin.isInitialized()) {
|
handleConnectionStateChange(connectionState)
|
||||||
when (state) {
|
|
||||||
ConnectionState.CONNECTED -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.CONNECTED))
|
|
||||||
ConnectionState.DISCONNECTED -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.DISCONNECTED))
|
|
||||||
ConnectionState.CONNECTING -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.CONNECTING))
|
|
||||||
ConnectionState.DISCONNECTING -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.DISCONNECTING))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -163,25 +156,31 @@ class MedtrumService : DaggerService(), BLECommCallback {
|
||||||
|
|
||||||
fun connect(from: String): Boolean {
|
fun connect(from: String): Boolean {
|
||||||
aapsLogger.debug(LTag.PUMP, "connect: called from: $from")
|
aapsLogger.debug(LTag.PUMP, "connect: called from: $from")
|
||||||
if (currentState is IdleState) {
|
when (currentState) {
|
||||||
|
is IdleState -> {
|
||||||
medtrumPump.connectionState = ConnectionState.CONNECTING
|
medtrumPump.connectionState = ConnectionState.CONNECTING
|
||||||
return bleComm.connect(from, medtrumPump.pumpSN)
|
return bleComm.connect(from, medtrumPump.pumpSN)
|
||||||
} else if (currentState is ReadyState) {
|
}
|
||||||
|
|
||||||
|
is ReadyState -> {
|
||||||
aapsLogger.error(LTag.PUMPCOMM, "Connect attempt when in ReadyState from: $from")
|
aapsLogger.error(LTag.PUMPCOMM, "Connect attempt when in ReadyState from: $from")
|
||||||
if (isConnected) {
|
return if (isConnected) {
|
||||||
aapsLogger.debug(LTag.PUMP, "connect: already connected")
|
aapsLogger.debug(LTag.PUMP, "connect: already connected")
|
||||||
return true
|
true
|
||||||
} else {
|
} else {
|
||||||
aapsLogger.debug(LTag.PUMP, "connect: not connected, resetting state and trying to connect")
|
aapsLogger.debug(LTag.PUMP, "connect: not connected, resetting state and trying to connect")
|
||||||
toState(IdleState())
|
toState(IdleState())
|
||||||
medtrumPump.connectionState = ConnectionState.CONNECTING
|
medtrumPump.connectionState = ConnectionState.CONNECTING
|
||||||
return bleComm.connect(from, medtrumPump.pumpSN)
|
bleComm.connect(from, medtrumPump.pumpSN)
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
aapsLogger.error(LTag.PUMPCOMM, "Connect attempt when in state: $currentState from: $from")
|
aapsLogger.error(LTag.PUMPCOMM, "Connect attempt when in state: $currentState from: $from")
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fun startPrime(): Boolean {
|
fun startPrime(): Boolean {
|
||||||
return sendPacketAndGetResponse(PrimePacket(injector))
|
return sendPacketAndGetResponse(PrimePacket(injector))
|
||||||
|
@ -247,8 +246,8 @@ class MedtrumService : DaggerService(), BLECommCallback {
|
||||||
timeUpdateNotification(result)
|
timeUpdateNotification(result)
|
||||||
}
|
}
|
||||||
// Do this here, because TBR can be cancelled due to time change by connect flow
|
// Do this here, because TBR can be cancelled due to time change by connect flow
|
||||||
if (needLoadHistory) {
|
if (needLoadHistory && result) {
|
||||||
if (result) result = loadEvents()
|
result = loadEvents()
|
||||||
}
|
}
|
||||||
if (result) medtrumPump.needCheckTimeUpdate = false
|
if (result) medtrumPump.needCheckTimeUpdate = false
|
||||||
return result
|
return result
|
||||||
|
@ -606,6 +605,17 @@ class MedtrumService : DaggerService(), BLECommCallback {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun handleConnectionStateChange(connectionState: ConnectionState) {
|
||||||
|
if (medtrumPlugin.isInitialized()) {
|
||||||
|
when (connectionState) {
|
||||||
|
ConnectionState.CONNECTED -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.CONNECTED))
|
||||||
|
ConnectionState.DISCONNECTED -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.DISCONNECTED))
|
||||||
|
ConnectionState.CONNECTING -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.CONNECTING))
|
||||||
|
ConnectionState.DISCONNECTING -> rxBus.send(EventPumpStatusChanged(EventPumpStatusChanged.Status.DISCONNECTING))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** BLECommCallbacks */
|
/** BLECommCallbacks */
|
||||||
override fun onBLEConnected() {
|
override fun onBLEConnected() {
|
||||||
aapsLogger.debug(LTag.PUMPCOMM, "<<<<< onBLEConnected")
|
aapsLogger.debug(LTag.PUMPCOMM, "<<<<< onBLEConnected")
|
||||||
|
@ -676,7 +686,10 @@ class MedtrumService : DaggerService(), BLECommCallback {
|
||||||
protected var responseSuccess = false
|
protected var responseSuccess = false
|
||||||
protected var sendRetryCounter = 0
|
protected var sendRetryCounter = 0
|
||||||
|
|
||||||
open fun onEnter() {}
|
open fun onEnter() {
|
||||||
|
// Used when a state is entered
|
||||||
|
}
|
||||||
|
|
||||||
open fun onIndication(data: ByteArray) {
|
open fun onIndication(data: ByteArray) {
|
||||||
aapsLogger.warn(LTag.PUMPCOMM, "onIndication: " + this.toString() + "Should not be called here!")
|
aapsLogger.warn(LTag.PUMPCOMM, "onIndication: " + this.toString() + "Should not be called here!")
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import info.nightscout.pump.medtrum.ui.event.SingleLiveEvent
|
||||||
import info.nightscout.pump.medtrum.ui.event.UIEvent
|
import info.nightscout.pump.medtrum.ui.event.UIEvent
|
||||||
import info.nightscout.interfaces.profile.ProfileFunction
|
import info.nightscout.interfaces.profile.ProfileFunction
|
||||||
import info.nightscout.interfaces.queue.CommandQueue
|
import info.nightscout.interfaces.queue.CommandQueue
|
||||||
|
import info.nightscout.pump.medtrum.MedtrumPlugin
|
||||||
import info.nightscout.pump.medtrum.MedtrumPump
|
import info.nightscout.pump.medtrum.MedtrumPump
|
||||||
import info.nightscout.pump.medtrum.R
|
import info.nightscout.pump.medtrum.R
|
||||||
import info.nightscout.pump.medtrum.code.ConnectionState
|
import info.nightscout.pump.medtrum.code.ConnectionState
|
||||||
|
@ -28,6 +29,7 @@ class MedtrumOverviewViewModel @Inject constructor(
|
||||||
private val profileFunction: ProfileFunction,
|
private val profileFunction: ProfileFunction,
|
||||||
private val commandQueue: CommandQueue,
|
private val commandQueue: CommandQueue,
|
||||||
private val dateUtil: DateUtil,
|
private val dateUtil: DateUtil,
|
||||||
|
private val medtrumPlugin: MedtrumPlugin,
|
||||||
val medtrumPump: MedtrumPump
|
val medtrumPump: MedtrumPump
|
||||||
) : BaseViewModel<MedtrumBaseNavigator>() {
|
) : BaseViewModel<MedtrumBaseNavigator>() {
|
||||||
|
|
||||||
|
@ -128,7 +130,7 @@ class MedtrumOverviewViewModel @Inject constructor(
|
||||||
scope.launch {
|
scope.launch {
|
||||||
medtrumPump.bolusAmountDeliveredFlow.collect { bolusAmount ->
|
medtrumPump.bolusAmountDeliveredFlow.collect { bolusAmount ->
|
||||||
aapsLogger.debug(LTag.PUMP, "MedtrumViewModel bolusAmountDeliveredFlow: $bolusAmount")
|
aapsLogger.debug(LTag.PUMP, "MedtrumViewModel bolusAmountDeliveredFlow: $bolusAmount")
|
||||||
if (!medtrumPump.bolusDone) {
|
if (!medtrumPump.bolusDone && medtrumPlugin.isInitialized()) {
|
||||||
_activeBolusStatus.postValue(
|
_activeBolusStatus.postValue(
|
||||||
dateUtil.timeString(medtrumPump.bolusStartTime) + " " + dateUtil.sinceString(medtrumPump.bolusStartTime, rh)
|
dateUtil.timeString(medtrumPump.bolusStartTime) + " " + dateUtil.sinceString(medtrumPump.bolusStartTime, rh)
|
||||||
+ " " + rh.gs(info.nightscout.interfaces.R.string.format_insulin_units, bolusAmount) + " / " + rh.gs(
|
+ " " + rh.gs(info.nightscout.interfaces.R.string.format_insulin_units, bolusAmount) + " / " + rh.gs(
|
||||||
|
@ -145,6 +147,8 @@ class MedtrumOverviewViewModel @Inject constructor(
|
||||||
kotlinx.coroutines.delay(T.mins(1).msecs())
|
kotlinx.coroutines.delay(T.mins(1).msecs())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Update gui on init
|
||||||
|
updateGUI()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onCleared() {
|
override fun onCleared() {
|
||||||
|
@ -189,7 +193,7 @@ class MedtrumOverviewViewModel @Inject constructor(
|
||||||
)
|
)
|
||||||
else _lastBolus.postValue("")
|
else _lastBolus.postValue("")
|
||||||
}
|
}
|
||||||
if (medtrumPump.bolusDone) {
|
if (medtrumPump.bolusDone || !medtrumPlugin.isInitialized()) {
|
||||||
_activeBolusStatus.postValue("")
|
_activeBolusStatus.postValue("")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -80,6 +80,7 @@ class MedtrumViewModel @Inject constructor(
|
||||||
}
|
}
|
||||||
|
|
||||||
ConnectionState.CONNECTING, ConnectionState.DISCONNECTING -> {
|
ConnectionState.CONNECTING, ConnectionState.DISCONNECTING -> {
|
||||||
|
// Do nothing
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,7 @@ internal annotation class ViewModelKey(val value: KClass<out ViewModel>)
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
class ViewModelFactory @Inject constructor(private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>) : ViewModelProvider.Factory {
|
class ViewModelFactory @Inject constructor(private val creators: Map<Class<out ViewModel>, @JvmSuppressWildcards Provider<ViewModel>>) : ViewModelProvider.Factory {
|
||||||
|
|
||||||
override fun <T : ViewModel> create(modelClass: Class<T>): T {
|
override fun <T : ViewModel> create(modelClass: Class<T>): T {
|
||||||
var creator: Provider<out ViewModel>? = creators[modelClass]
|
var creator: Provider<out ViewModel>? = creators[modelClass]
|
||||||
if (creator == null) {
|
if (creator == null) {
|
||||||
|
@ -26,9 +27,7 @@ class ViewModelFactory @Inject constructor(private val creators: Map<Class<out V
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (creator == null) {
|
requireNotNull(creator) { "unknown model class $modelClass" }
|
||||||
throw IllegalArgumentException("unknown model class $modelClass")
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
@Suppress("UNCHECKED_CAST")
|
@Suppress("UNCHECKED_CAST")
|
||||||
return creator.get() as T
|
return creator.get() as T
|
||||||
|
|
|
@ -25,7 +25,7 @@
|
||||||
|
|
||||||
<string name="medtrum">Medtrum</string>
|
<string name="medtrum">Medtrum</string>
|
||||||
<string name="medtrum_pump_shortname">MT</string>
|
<string name="medtrum_pump_shortname">MT</string>
|
||||||
<string name="medtrum_pump_description">Pump integration for Medtrum Nano</string>
|
<string name="medtrum_pump_description">Pump integration for Medtrum Nano and Medtrum 300U</string>
|
||||||
<string name="medtrum_pump_setting">Medtrum pump settings</string>
|
<string name="medtrum_pump_setting">Medtrum pump settings</string>
|
||||||
<string name="pump_error">Pump error: %1$s !! </string>
|
<string name="pump_error">Pump error: %1$s !! </string>
|
||||||
<string name="pump_is_suspended">Pump is suspended</string>
|
<string name="pump_is_suspended">Pump is suspended</string>
|
||||||
|
|
|
@ -44,7 +44,7 @@
|
||||||
android:title="@string/daily_max_insulin_title"
|
android:title="@string/daily_max_insulin_title"
|
||||||
android:dialogMessage="@string/daily_max_insulin_summary"
|
android:dialogMessage="@string/daily_max_insulin_summary"
|
||||||
validate:maxNumber="180"
|
validate:maxNumber="180"
|
||||||
validate:minNumber="25"
|
validate:minNumber="20"
|
||||||
validate:testType="numericRange" />
|
validate:testType="numericRange" />
|
||||||
|
|
||||||
</PreferenceCategory>
|
</PreferenceCategory>
|
||||||
|
|
Loading…
Reference in a new issue