2019-12-20 18:55:54 +01:00
|
|
|
package info.nightscout.androidaps.dialogs
|
2019-12-20 15:36:27 +01:00
|
|
|
|
2020-01-12 23:43:44 +01:00
|
|
|
import android.content.Context
|
2019-12-20 15:36:27 +01:00
|
|
|
import android.os.Bundle
|
|
|
|
import android.text.Editable
|
|
|
|
import android.text.TextWatcher
|
|
|
|
import android.view.LayoutInflater
|
|
|
|
import android.view.View
|
|
|
|
import android.view.ViewGroup
|
|
|
|
import com.google.common.base.Joiner
|
2021-04-14 18:42:12 +02:00
|
|
|
import info.nightscout.androidaps.interfaces.Config
|
2019-12-20 15:36:27 +01:00
|
|
|
import info.nightscout.androidaps.R
|
2019-12-20 23:05:35 +01:00
|
|
|
import info.nightscout.androidaps.activities.ErrorHelperActivity
|
2019-12-20 15:36:27 +01:00
|
|
|
import info.nightscout.androidaps.data.DetailedBolusInfo
|
2021-03-25 17:48:07 +01:00
|
|
|
import info.nightscout.androidaps.database.AppRepository
|
2021-04-03 21:29:26 +02:00
|
|
|
import info.nightscout.androidaps.database.entities.ValueWithUnit
|
2021-03-22 16:59:09 +01:00
|
|
|
import info.nightscout.androidaps.database.entities.UserEntry.Action
|
2021-03-31 00:18:55 +02:00
|
|
|
import info.nightscout.androidaps.database.entities.UserEntry.Sources
|
2021-01-21 20:46:21 +01:00
|
|
|
import info.nightscout.androidaps.databinding.DialogTreatmentBinding
|
2021-04-14 00:45:30 +02:00
|
|
|
import info.nightscout.androidaps.interfaces.ActivePlugin
|
2021-11-07 17:19:06 +01:00
|
|
|
import info.nightscout.androidaps.interfaces.CommandQueue
|
2019-12-20 15:36:27 +01:00
|
|
|
import info.nightscout.androidaps.interfaces.Constraint
|
2021-12-10 15:19:19 +01:00
|
|
|
import info.nightscout.shared.logging.LTag
|
2021-02-09 17:57:28 +01:00
|
|
|
import info.nightscout.androidaps.logging.UserEntryLogger
|
2019-12-27 19:20:38 +01:00
|
|
|
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
|
2019-12-20 15:36:27 +01:00
|
|
|
import info.nightscout.androidaps.queue.Callback
|
2019-12-20 23:05:35 +01:00
|
|
|
import info.nightscout.androidaps.utils.DecimalFormatter
|
|
|
|
import info.nightscout.androidaps.utils.HtmlHelper
|
2021-12-10 08:58:23 +01:00
|
|
|
import info.nightscout.shared.SafeParse
|
2019-12-20 23:05:35 +01:00
|
|
|
import info.nightscout.androidaps.utils.ToastUtils
|
2021-01-21 20:46:21 +01:00
|
|
|
import info.nightscout.androidaps.utils.alertDialogs.OKDialog
|
2021-04-05 22:41:28 +02:00
|
|
|
import info.nightscout.androidaps.extensions.formatColor
|
2022-03-07 11:49:37 +01:00
|
|
|
import info.nightscout.androidaps.utils.protection.ProtectionCheck
|
2022-03-23 13:39:38 +01:00
|
|
|
import info.nightscout.androidaps.utils.protection.ProtectionCheck.Protection.BOLUS
|
2022-04-23 18:34:04 +02:00
|
|
|
import info.nightscout.androidaps.interfaces.ResourceHelper
|
2022-02-10 19:11:58 +01:00
|
|
|
import io.reactivex.rxjava3.disposables.CompositeDisposable
|
|
|
|
import io.reactivex.rxjava3.kotlin.plusAssign
|
2019-12-20 15:36:27 +01:00
|
|
|
import java.text.DecimalFormat
|
|
|
|
import java.util.*
|
2019-12-27 19:20:38 +01:00
|
|
|
import javax.inject.Inject
|
2019-12-20 15:36:27 +01:00
|
|
|
import kotlin.math.abs
|
|
|
|
|
|
|
|
class TreatmentDialog : DialogFragmentWithDate() {
|
2021-01-21 20:46:21 +01:00
|
|
|
|
2019-12-31 00:37:36 +01:00
|
|
|
@Inject lateinit var constraintChecker: ConstraintChecker
|
2021-11-04 10:56:12 +01:00
|
|
|
@Inject lateinit var rh: ResourceHelper
|
2021-04-14 00:45:30 +02:00
|
|
|
@Inject lateinit var activePlugin: ActivePlugin
|
2021-11-07 17:19:06 +01:00
|
|
|
@Inject lateinit var commandQueue: CommandQueue
|
2020-01-12 23:43:44 +01:00
|
|
|
@Inject lateinit var ctx: Context
|
2020-07-24 13:00:11 +02:00
|
|
|
@Inject lateinit var config: Config
|
2021-02-09 17:57:28 +01:00
|
|
|
@Inject lateinit var uel: UserEntryLogger
|
2021-03-25 17:48:07 +01:00
|
|
|
@Inject lateinit var repository: AppRepository
|
2022-03-07 11:49:37 +01:00
|
|
|
@Inject lateinit var protectionCheck: ProtectionCheck
|
2021-03-25 17:48:07 +01:00
|
|
|
|
2022-03-23 13:39:38 +01:00
|
|
|
private var queryingProtection = false
|
2021-03-25 17:48:07 +01:00
|
|
|
private val disposable = CompositeDisposable()
|
2022-03-23 13:39:38 +01:00
|
|
|
private var _binding: DialogTreatmentBinding? = null
|
|
|
|
|
|
|
|
// This property is only valid between onCreateView and onDestroyView.
|
|
|
|
private val binding get() = _binding!!
|
2019-12-20 15:36:27 +01:00
|
|
|
|
|
|
|
private val textWatcher: TextWatcher = object : TextWatcher {
|
|
|
|
override fun afterTextChanged(s: Editable) {}
|
|
|
|
override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {}
|
|
|
|
override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {
|
|
|
|
validateInputs()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun validateInputs() {
|
2019-12-27 19:20:38 +01:00
|
|
|
val maxCarbs = constraintChecker.getMaxCarbsAllowed().value().toDouble()
|
|
|
|
val maxInsulin = constraintChecker.getMaxBolusAllowed().value()
|
2021-01-21 20:46:21 +01:00
|
|
|
if (SafeParse.stringToInt(binding.carbs.text) > maxCarbs) {
|
|
|
|
binding.carbs.value = 0.0
|
2021-11-04 10:56:12 +01:00
|
|
|
ToastUtils.showToastInUiThread(context, rh.gs(R.string.carbsconstraintapplied))
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
2021-01-21 20:46:21 +01:00
|
|
|
if (SafeParse.stringToDouble(binding.insulin.text) > maxInsulin) {
|
|
|
|
binding.insulin.value = 0.0
|
2021-11-04 10:56:12 +01:00
|
|
|
ToastUtils.showToastInUiThread(context, rh.gs(R.string.bolusconstraintapplied))
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun onSaveInstanceState(savedInstanceState: Bundle) {
|
|
|
|
super.onSaveInstanceState(savedInstanceState)
|
2021-01-21 20:46:21 +01:00
|
|
|
savedInstanceState.putDouble("carbs", binding.carbs.value)
|
|
|
|
savedInstanceState.putDouble("insulin", binding.insulin.value)
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
|
2021-01-21 20:46:21 +01:00
|
|
|
savedInstanceState: Bundle?): View {
|
2019-12-20 23:05:35 +01:00
|
|
|
onCreateViewGeneral()
|
2021-01-21 20:46:21 +01:00
|
|
|
_binding = DialogTreatmentBinding.inflate(inflater, container, false)
|
|
|
|
return binding.root
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
|
|
|
super.onViewCreated(view, savedInstanceState)
|
|
|
|
|
2020-07-24 13:00:11 +02:00
|
|
|
if (config.NSCLIENT) {
|
2021-01-21 20:46:21 +01:00
|
|
|
binding.recordOnly.isChecked = true
|
|
|
|
binding.recordOnly.isEnabled = false
|
2020-07-24 13:00:11 +02:00
|
|
|
}
|
2019-12-27 19:20:38 +01:00
|
|
|
val maxCarbs = constraintChecker.getMaxCarbsAllowed().value().toDouble()
|
|
|
|
val maxInsulin = constraintChecker.getMaxBolusAllowed().value()
|
2020-03-16 21:40:29 +01:00
|
|
|
val pumpDescription = activePlugin.activePump.pumpDescription
|
2021-01-21 20:46:21 +01:00
|
|
|
binding.carbs.setParams(savedInstanceState?.getDouble("carbs")
|
|
|
|
?: 0.0, 0.0, maxCarbs, 1.0, DecimalFormat("0"), false, binding.okcancel.ok, textWatcher)
|
|
|
|
binding.insulin.setParams(savedInstanceState?.getDouble("insulin")
|
|
|
|
?: 0.0, 0.0, maxInsulin, pumpDescription.bolusStep, DecimalFormatter.pumpSupportedBolusFormat(activePlugin.activePump), false, binding.okcancel.ok, textWatcher)
|
2021-11-25 19:26:43 +01:00
|
|
|
binding.recordOnlyLayout.visibility = View.GONE
|
2022-04-11 14:22:12 +02:00
|
|
|
binding.insulinLabel.labelFor = binding.insulin.editTextId
|
|
|
|
binding.carbsLabel.labelFor = binding.carbs.editTextId
|
2021-01-21 20:46:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
override fun onDestroyView() {
|
|
|
|
super.onDestroyView()
|
|
|
|
_binding = null
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
|
2019-12-21 23:17:20 +01:00
|
|
|
override fun submit(): Boolean {
|
2021-01-21 20:46:21 +01:00
|
|
|
if (_binding == null) return false
|
2020-03-16 21:40:29 +01:00
|
|
|
val pumpDescription = activePlugin.activePump.pumpDescription
|
2021-11-23 19:53:45 +01:00
|
|
|
val insulin = SafeParse.stringToDouble(binding.insulin.text)
|
2021-01-21 20:46:21 +01:00
|
|
|
val carbs = SafeParse.stringToInt(binding.carbs.text)
|
|
|
|
val recordOnlyChecked = binding.recordOnly.isChecked
|
2019-12-20 15:36:27 +01:00
|
|
|
val actions: LinkedList<String?> = LinkedList()
|
2019-12-27 19:20:38 +01:00
|
|
|
val insulinAfterConstraints = constraintChecker.applyBolusConstraints(Constraint(insulin)).value()
|
|
|
|
val carbsAfterConstraints = constraintChecker.applyCarbsConstraints(Constraint(carbs)).value()
|
2019-12-20 15:36:27 +01:00
|
|
|
|
|
|
|
if (insulinAfterConstraints > 0) {
|
2022-03-29 17:39:20 +02:00
|
|
|
actions.add(rh.gs(R.string.bolus) + ": " + DecimalFormatter.toPumpSupportedBolus(insulinAfterConstraints, activePlugin.activePump, rh).formatColor(context, rh, R.attr.bolusColor))
|
2019-12-20 15:36:27 +01:00
|
|
|
if (recordOnlyChecked)
|
2022-03-29 17:39:20 +02:00
|
|
|
actions.add(rh.gs(R.string.bolusrecordedonly).formatColor(context, rh, R.attr.warningColor))
|
2019-12-20 15:36:27 +01:00
|
|
|
if (abs(insulinAfterConstraints - insulin) > pumpDescription.pumpType.determineCorrectBolusStepSize(insulinAfterConstraints))
|
2022-03-29 17:39:20 +02:00
|
|
|
actions.add(rh.gs(R.string.bolusconstraintappliedwarn, insulin, insulinAfterConstraints).formatColor(context, rh, R.attr.warningColor))
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
if (carbsAfterConstraints > 0) {
|
2022-03-29 17:39:20 +02:00
|
|
|
actions.add(rh.gs(R.string.carbs) + ": " + rh.gs(R.string.format_carbs, carbsAfterConstraints).formatColor(context, rh, R.attr.carbsColor))
|
2019-12-20 15:36:27 +01:00
|
|
|
if (carbsAfterConstraints != carbs)
|
2022-03-29 17:39:20 +02:00
|
|
|
actions.add(rh.gs(R.string.carbsconstraintapplied).formatColor(context, rh, R.attr.warningColor))
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
if (insulinAfterConstraints > 0 || carbsAfterConstraints > 0) {
|
|
|
|
activity?.let { activity ->
|
2021-11-04 10:56:12 +01:00
|
|
|
OKDialog.showConfirmation(activity, rh.gs(R.string.overview_treatment_label), HtmlHelper.fromHtml(Joiner.on("<br/>").join(actions)), {
|
2021-03-25 13:32:11 +01:00
|
|
|
val action = when {
|
|
|
|
insulinAfterConstraints.equals(0.0) -> Action.CARBS
|
2021-11-23 19:53:45 +01:00
|
|
|
carbsAfterConstraints == 0 -> Action.BOLUS
|
2021-03-25 13:32:11 +01:00
|
|
|
else -> Action.TREATMENT
|
|
|
|
}
|
2019-12-20 15:36:27 +01:00
|
|
|
val detailedBolusInfo = DetailedBolusInfo()
|
2021-03-25 17:48:07 +01:00
|
|
|
if (insulinAfterConstraints == 0.0) detailedBolusInfo.eventType = DetailedBolusInfo.EventType.CARBS_CORRECTION
|
|
|
|
if (carbsAfterConstraints == 0) detailedBolusInfo.eventType = DetailedBolusInfo.EventType.CORRECTION_BOLUS
|
2019-12-20 15:36:27 +01:00
|
|
|
detailedBolusInfo.insulin = insulinAfterConstraints
|
|
|
|
detailedBolusInfo.carbs = carbsAfterConstraints.toDouble()
|
|
|
|
detailedBolusInfo.context = context
|
2021-03-25 23:52:08 +01:00
|
|
|
if (recordOnlyChecked) {
|
2021-11-04 10:56:12 +01:00
|
|
|
uel.log(action, Sources.TreatmentDialog, if (insulinAfterConstraints != 0.0) rh.gs(R.string.record) else "",
|
2021-04-03 21:29:26 +02:00
|
|
|
ValueWithUnit.Timestamp(detailedBolusInfo.timestamp).takeIf { eventTimeChanged },
|
2021-11-04 10:56:12 +01:00
|
|
|
ValueWithUnit.SimpleString(rh.gsNotLocalised(R.string.record)).takeIf { insulinAfterConstraints != 0.0 },
|
2021-04-03 21:29:26 +02:00
|
|
|
ValueWithUnit.Insulin(insulinAfterConstraints).takeIf { insulinAfterConstraints != 0.0 },
|
|
|
|
ValueWithUnit.Gram(carbsAfterConstraints).takeIf { carbsAfterConstraints != 0 })
|
2021-03-29 19:15:46 +02:00
|
|
|
if (detailedBolusInfo.insulin > 0)
|
|
|
|
disposable += repository.runTransactionForResult(detailedBolusInfo.insertBolusTransaction())
|
|
|
|
.subscribe(
|
|
|
|
{ result -> result.inserted.forEach { aapsLogger.debug(LTag.DATABASE, "Inserted bolus $it") } },
|
|
|
|
{ aapsLogger.error(LTag.DATABASE, "Error while saving bolus", it) }
|
|
|
|
)
|
|
|
|
if (detailedBolusInfo.carbs > 0)
|
|
|
|
disposable += repository.runTransactionForResult(detailedBolusInfo.insertCarbsTransaction())
|
|
|
|
.subscribe(
|
|
|
|
{ result -> result.inserted.forEach { aapsLogger.debug(LTag.DATABASE, "Inserted carbs $it") } },
|
|
|
|
{ aapsLogger.error(LTag.DATABASE, "Error while saving carbs", it) }
|
|
|
|
)
|
2021-03-25 23:52:08 +01:00
|
|
|
} else {
|
2021-04-10 19:56:28 +02:00
|
|
|
if (detailedBolusInfo.insulin > 0) {
|
|
|
|
uel.log(action, Sources.TreatmentDialog,
|
|
|
|
ValueWithUnit.Insulin(insulinAfterConstraints),
|
|
|
|
ValueWithUnit.Gram(carbsAfterConstraints).takeIf { carbsAfterConstraints != 0 })
|
2021-04-01 22:23:53 +02:00
|
|
|
commandQueue.bolus(detailedBolusInfo, object : Callback() {
|
|
|
|
override fun run() {
|
|
|
|
if (!result.success) {
|
2021-11-04 10:56:12 +01:00
|
|
|
ErrorHelperActivity.runAlarm(ctx, result.comment, rh.gs(R.string.treatmentdeliveryerror), R.raw.boluserror)
|
2021-04-10 19:56:28 +02:00
|
|
|
}
|
2021-03-25 23:52:08 +01:00
|
|
|
}
|
2021-04-01 22:23:53 +02:00
|
|
|
})
|
2022-02-03 21:33:56 +01:00
|
|
|
} else {
|
2021-04-01 22:23:53 +02:00
|
|
|
uel.log(action, Sources.TreatmentDialog,
|
2022-02-03 21:33:56 +01:00
|
|
|
ValueWithUnit.Gram(carbsAfterConstraints).takeIf { carbsAfterConstraints != 0 })
|
|
|
|
if (detailedBolusInfo.carbs > 0) {
|
|
|
|
disposable += repository.runTransactionForResult(detailedBolusInfo.insertCarbsTransaction())
|
|
|
|
.subscribe(
|
|
|
|
{ result -> result.inserted.forEach { aapsLogger.debug(LTag.DATABASE, "Inserted carbs $it") } },
|
|
|
|
{ aapsLogger.error(LTag.DATABASE, "Error while saving carbs", it) }
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2021-03-25 17:48:07 +01:00
|
|
|
}
|
2019-12-22 21:37:26 +01:00
|
|
|
})
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
|
|
|
} else
|
2019-12-22 21:37:26 +01:00
|
|
|
activity?.let { activity ->
|
2021-11-04 10:56:12 +01:00
|
|
|
OKDialog.show(activity, rh.gs(R.string.overview_treatment_label), rh.gs(R.string.no_action_selected))
|
2019-12-22 21:37:26 +01:00
|
|
|
}
|
2019-12-21 23:17:20 +01:00
|
|
|
return true
|
2019-12-20 15:36:27 +01:00
|
|
|
}
|
2022-03-07 11:49:37 +01:00
|
|
|
|
|
|
|
override fun onResume() {
|
|
|
|
super.onResume()
|
2022-03-23 13:39:38 +01:00
|
|
|
if(!queryingProtection) {
|
|
|
|
queryingProtection = true
|
|
|
|
activity?.let { activity ->
|
|
|
|
val cancelFail = {
|
|
|
|
queryingProtection = false
|
|
|
|
aapsLogger.debug(LTag.APS, "Dialog canceled on resume protection: ${this.javaClass.name}")
|
|
|
|
ToastUtils.showToastInUiThread(ctx, R.string.dialog_canceled)
|
|
|
|
dismiss()
|
|
|
|
}
|
|
|
|
protectionCheck.queryProtection(activity, BOLUS, { queryingProtection = false }, cancelFail, cancelFail)
|
2022-03-07 11:49:37 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|