AndroidAPS/app/src/main/java/info/nightscout/androidaps/dialogs/InsulinDialog.kt

237 lines
12 KiB
Kotlin
Raw Normal View History

2019-12-20 18:55:54 +01:00
package info.nightscout.androidaps.dialogs
2019-12-20 14:52:10 +01:00
2020-01-12 23:43:44 +01:00
import android.content.Context
2019-12-20 14:52:10 +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
2020-07-24 13:00:11 +02:00
import info.nightscout.androidaps.Config
2019-12-20 14:52:10 +01:00
import info.nightscout.androidaps.Constants
import info.nightscout.androidaps.R
2019-12-20 23:05:35 +01:00
import info.nightscout.androidaps.activities.ErrorHelperActivity
2019-12-20 14:52:10 +01:00
import info.nightscout.androidaps.data.DetailedBolusInfo
import info.nightscout.androidaps.data.Profile
2021-02-09 17:57:28 +01:00
import info.nightscout.androidaps.database.AppRepository
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.database.entities.TemporaryTarget
2021-03-08 20:10:02 +01:00
import info.nightscout.androidaps.database.entities.TherapyEvent
2021-03-22 16:59:09 +01:00
import info.nightscout.androidaps.database.entities.UserEntry.Action
import info.nightscout.androidaps.database.entities.UserEntry.Units
import info.nightscout.androidaps.database.entities.UserEntry.ValueWithUnit
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.database.transactions.InsertTemporaryTargetAndCancelCurrentTransaction
2021-01-21 20:31:20 +01:00
import info.nightscout.androidaps.databinding.DialogInsulinBinding
2020-01-10 23:14:58 +01:00
import info.nightscout.androidaps.interfaces.ActivePluginProvider
import info.nightscout.androidaps.interfaces.CommandQueueProvider
2019-12-20 14:52:10 +01:00
import info.nightscout.androidaps.interfaces.Constraint
2020-05-07 09:54:36 +02:00
import info.nightscout.androidaps.interfaces.ProfileFunction
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.logging.LTag
2021-02-09 17:57:28 +01:00
import info.nightscout.androidaps.logging.UserEntryLogger
2021-01-21 20:31:20 +01:00
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.plugins.general.nsclient.NSUpload
2019-12-20 14:52:10 +01:00
import info.nightscout.androidaps.queue.Callback
import info.nightscout.androidaps.utils.*
2020-04-08 14:03:57 +02:00
import info.nightscout.androidaps.utils.alertDialogs.OKDialog
2020-08-19 21:24:01 +02:00
import info.nightscout.androidaps.utils.extensions.formatColor
2019-12-31 11:57:58 +01:00
import info.nightscout.androidaps.utils.extensions.toSignedString
2020-04-08 14:03:57 +02:00
import info.nightscout.androidaps.utils.extensions.toVisibility
2019-12-27 19:20:38 +01:00
import info.nightscout.androidaps.utils.resources.ResourceHelper
2021-03-01 12:42:42 +01:00
import io.reactivex.disposables.CompositeDisposable
import io.reactivex.rxkotlin.plusAssign
2019-12-20 14:52:10 +01:00
import java.text.DecimalFormat
import java.util.*
2021-03-01 12:42:42 +01:00
import java.util.concurrent.TimeUnit
2019-12-27 19:20:38 +01:00
import javax.inject.Inject
2019-12-20 14:52:10 +01:00
import kotlin.math.abs
import kotlin.math.max
class InsulinDialog : DialogFragmentWithDate() {
2021-01-21 20:31:20 +01:00
2019-12-30 00:53:44 +01:00
@Inject lateinit var constraintChecker: ConstraintChecker
@Inject lateinit var resourceHelper: ResourceHelper
2020-01-01 23:23:16 +01:00
@Inject lateinit var defaultValueHelper: DefaultValueHelper
2019-12-30 00:53:44 +01:00
@Inject lateinit var profileFunction: ProfileFunction
2020-01-10 23:14:58 +01:00
@Inject lateinit var commandQueue: CommandQueueProvider
@Inject lateinit var activePlugin: ActivePluginProvider
2020-01-12 23:43:44 +01:00
@Inject lateinit var ctx: Context
2021-02-09 17:57:28 +01:00
@Inject lateinit var repository: AppRepository
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-01 12:42:42 +01:00
@Inject lateinit var nsUpload: NSUpload
2019-12-27 19:20:38 +01:00
2019-12-20 14:52:10 +01:00
companion object {
2021-01-21 20:31:20 +01:00
2019-12-20 14:52:10 +01:00
private const val PLUS1_DEFAULT = 0.5
private const val PLUS2_DEFAULT = 1.0
private const val PLUS3_DEFAULT = 2.0
}
2021-03-01 12:42:42 +01:00
private val disposable = CompositeDisposable()
2019-12-20 14:52:10 +01:00
private val textWatcher: TextWatcher = object : TextWatcher {
override fun afterTextChanged(s: Editable) {
validateInputs()
}
override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {}
override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {}
}
2021-01-21 20:31:20 +01:00
private var _binding: DialogInsulinBinding? = null
// This property is only valid between onCreateView and
// onDestroyView.
private val binding get() = _binding!!
2019-12-20 14:52:10 +01:00
private fun validateInputs() {
2019-12-27 19:20:38 +01:00
val maxInsulin = constraintChecker.getMaxBolusAllowed().value()
2021-01-21 20:31:20 +01:00
if (abs(binding.time.value.toInt()) > 12 * 60) {
binding.time.value = 0.0
2020-01-10 23:14:58 +01:00
ToastUtils.showToastInUiThread(context, resourceHelper.gs(R.string.constraintapllied))
2019-12-20 14:52:10 +01:00
}
2021-01-21 20:31:20 +01:00
if (binding.amount.value > maxInsulin) {
binding.amount.value = 0.0
2020-01-10 23:14:58 +01:00
ToastUtils.showToastInUiThread(context, resourceHelper.gs(R.string.bolusconstraintapplied))
2019-12-20 14:52:10 +01:00
}
}
override fun onSaveInstanceState(savedInstanceState: Bundle) {
super.onSaveInstanceState(savedInstanceState)
2021-01-21 20:31:20 +01:00
savedInstanceState.putDouble("time", binding.time.value)
savedInstanceState.putDouble("amount", binding.amount.value)
2019-12-20 14:52:10 +01:00
}
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
2021-01-21 20:31:20 +01:00
savedInstanceState: Bundle?): View {
2019-12-20 23:05:35 +01:00
onCreateViewGeneral()
2021-01-21 20:31:20 +01:00
_binding = DialogInsulinBinding.inflate(inflater, container, false)
return binding.root
2019-12-20 14:52:10 +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:31:20 +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 maxInsulin = constraintChecker.getMaxBolusAllowed().value()
2021-01-21 20:31:20 +01:00
binding.time.setParams(savedInstanceState?.getDouble("time")
?: 0.0, -12 * 60.0, 12 * 60.0, 5.0, DecimalFormat("0"), false, binding.okcancel.ok, textWatcher)
binding.amount.setParams(savedInstanceState?.getDouble("amount")
?: 0.0, 0.0, maxInsulin, activePlugin.activePump.pumpDescription.bolusStep, DecimalFormatter.pumpSupportedBolusFormat(activePlugin.activePump), false, binding.okcancel.ok, textWatcher)
2019-12-20 14:52:10 +01:00
2021-01-21 20:31:20 +01:00
binding.plus05.text = sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_1), PLUS1_DEFAULT).toSignedString(activePlugin.activePump)
binding.plus05.setOnClickListener {
binding.amount.value = max(0.0, binding.amount.value
2019-12-30 00:53:44 +01:00
+ sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_1), PLUS1_DEFAULT))
2019-12-20 14:52:10 +01:00
validateInputs()
}
2021-01-21 20:31:20 +01:00
binding.plus10.text = sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_2), PLUS2_DEFAULT).toSignedString(activePlugin.activePump)
binding.plus10.setOnClickListener {
binding.amount.value = max(0.0, binding.amount.value
2019-12-30 00:53:44 +01:00
+ sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_2), PLUS2_DEFAULT))
2019-12-20 14:52:10 +01:00
validateInputs()
}
2021-01-21 20:31:20 +01:00
binding.plus20.text = sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_3), PLUS3_DEFAULT).toSignedString(activePlugin.activePump)
binding.plus20.setOnClickListener {
binding.amount.value = max(0.0, binding.amount.value
2019-12-30 00:53:44 +01:00
+ sp.getDouble(resourceHelper.gs(R.string.key_insulin_button_increment_3), PLUS3_DEFAULT))
2019-12-20 14:52:10 +01:00
validateInputs()
}
2021-01-21 20:31:20 +01:00
binding.timeLayout.visibility = View.GONE
binding.recordOnly.setOnCheckedChangeListener { _, isChecked: Boolean ->
binding.timeLayout.visibility = isChecked.toVisibility()
2019-12-20 14:52:10 +01:00
}
}
2021-01-21 20:31:20 +01:00
override fun onDestroyView() {
super.onDestroyView()
2021-03-01 12:42:42 +01:00
disposable.clear()
2021-01-21 20:31:20 +01:00
_binding = null
}
2019-12-21 23:17:20 +01:00
override fun submit(): Boolean {
2021-01-21 20:31:20 +01:00
if (_binding == null) return false
2020-03-16 21:40:29 +01:00
val pumpDescription = activePlugin.activePump.pumpDescription
2021-01-21 20:31:20 +01:00
val insulin = SafeParse.stringToDouble(binding.amount.text ?: return false)
2019-12-27 19:20:38 +01:00
val insulinAfterConstraints = constraintChecker.applyBolusConstraints(Constraint(insulin)).value()
2019-12-20 14:52:10 +01:00
val actions: LinkedList<String?> = LinkedList()
2019-12-27 19:20:38 +01:00
val units = profileFunction.getUnits()
val unitLabel = if (units == Constants.MMOL) resourceHelper.gs(R.string.mmol) else resourceHelper.gs(R.string.mgdl)
2021-01-21 20:31:20 +01:00
val recordOnlyChecked = binding.recordOnly.isChecked
val eatingSoonChecked = binding.startEatingSoonTt.isChecked
2019-12-20 14:52:10 +01:00
2019-12-20 15:36:27 +01:00
if (insulinAfterConstraints > 0) {
2020-08-19 21:24:01 +02:00
actions.add(resourceHelper.gs(R.string.bolus) + ": " + DecimalFormatter.toPumpSupportedBolus(insulinAfterConstraints, activePlugin.activePump, resourceHelper).formatColor(resourceHelper, R.color.bolus))
2019-12-20 14:52:10 +01:00
if (recordOnlyChecked)
2020-08-19 21:24:01 +02:00
actions.add(resourceHelper.gs(R.string.bolusrecordedonly).formatColor(resourceHelper, R.color.warning))
2019-12-20 15:36:27 +01:00
if (abs(insulinAfterConstraints - insulin) > pumpDescription.pumpType.determineCorrectBolusStepSize(insulinAfterConstraints))
2020-08-19 21:24:01 +02:00
actions.add(resourceHelper.gs(R.string.bolusconstraintappliedwarn, insulin, insulinAfterConstraints).formatColor(resourceHelper, R.color.warning))
2019-12-20 14:52:10 +01:00
}
2020-01-01 23:23:16 +01:00
val eatingSoonTTDuration = defaultValueHelper.determineEatingSoonTTDuration()
val eatingSoonTT = defaultValueHelper.determineEatingSoonTT()
2019-12-20 14:52:10 +01:00
if (eatingSoonChecked)
2020-08-19 21:24:01 +02:00
actions.add(resourceHelper.gs(R.string.temptargetshort) + ": " + (DecimalFormatter.to1Decimal(eatingSoonTT) + " " + unitLabel + " (" + resourceHelper.gs(R.string.format_mins, eatingSoonTTDuration) + ")").formatColor(resourceHelper, R.color.tempTargetConfirmation))
2019-12-20 14:52:10 +01:00
2021-01-21 20:31:20 +01:00
val timeOffset = binding.time.value.toInt()
2019-12-20 14:52:10 +01:00
val time = DateUtil.now() + T.mins(timeOffset.toLong()).msecs()
if (timeOffset != 0)
2020-05-07 23:40:59 +02:00
actions.add(resourceHelper.gs(R.string.time) + ": " + dateUtil.dateAndTimeString(time))
2019-12-20 14:52:10 +01:00
2021-01-21 20:31:20 +01:00
val notes = binding.notesLayout.notes.text.toString()
2019-12-20 14:52:10 +01:00
if (notes.isNotEmpty())
2021-02-22 18:04:30 +01:00
actions.add(resourceHelper.gs(R.string.notes_label) + ": " + notes)
2019-12-20 14:52:10 +01:00
if (insulinAfterConstraints > 0 || eatingSoonChecked) {
activity?.let { activity ->
2021-01-21 20:31:20 +01:00
OKDialog.showConfirmation(activity, resourceHelper.gs(R.string.bolus), HtmlHelper.fromHtml(Joiner.on("<br/>").join(actions)), {
2019-12-20 14:52:10 +01:00
if (eatingSoonChecked) {
uel.log(Action.TT, notes, ValueWithUnit(TemporaryTarget.Reason.EATING_SOON.text, Units.TherapyEvent), ValueWithUnit(eatingSoonTT, units), ValueWithUnit(eatingSoonTTDuration, Units.M))
2021-03-01 12:42:42 +01:00
disposable += repository.runTransactionForResult(InsertTemporaryTargetAndCancelCurrentTransaction(
timestamp = System.currentTimeMillis(),
duration = TimeUnit.MINUTES.toMillis(eatingSoonTTDuration.toLong()),
reason = TemporaryTarget.Reason.EATING_SOON,
lowTarget = Profile.toMgdl(eatingSoonTT, profileFunction.getUnits()),
highTarget = Profile.toMgdl(eatingSoonTT, profileFunction.getUnits())
)).subscribe({ result ->
result.inserted.forEach { nsUpload.uploadTempTarget(it) }
result.updated.forEach { nsUpload.updateTempTarget(it) }
}, {
aapsLogger.error(LTag.BGSOURCE, "Error while saving temporary target", it)
})
2019-12-20 14:52:10 +01:00
}
if (insulinAfterConstraints > 0) {
val detailedBolusInfo = DetailedBolusInfo()
2021-03-22 16:59:09 +01:00
detailedBolusInfo.eventType = TherapyEvent.Type.CORRECTION_BOLUS
2019-12-20 14:52:10 +01:00
detailedBolusInfo.insulin = insulinAfterConstraints
detailedBolusInfo.context = context
detailedBolusInfo.notes = notes
if (recordOnlyChecked) {
2021-03-22 16:59:09 +01:00
uel.log(Action.BOLUS_RECORD, notes, ValueWithUnit(insulinAfterConstraints, Units.U), ValueWithUnit(timeOffset, Units.M, timeOffset != 0))
detailedBolusInfo.timestamp = time
2020-01-10 23:14:58 +01:00
activePlugin.activeTreatments.addToHistoryTreatment(detailedBolusInfo, false)
2019-12-20 14:52:10 +01:00
} else {
uel.log(Action.BOLUS, notes, ValueWithUnit(insulinAfterConstraints, Units.U))
2021-03-22 16:59:09 +01:00
detailedBolusInfo.timestamp = DateUtil.now()
2020-01-10 23:14:58 +01:00
commandQueue.bolus(detailedBolusInfo, object : Callback() {
2019-12-20 14:52:10 +01:00
override fun run() {
if (!result.success) {
2021-02-14 15:09:06 +01:00
ErrorHelperActivity.runAlarm(ctx, result.comment, resourceHelper.gs(R.string.treatmentdeliveryerror), info.nightscout.androidaps.dana.R.raw.boluserror)
2019-12-20 14:52:10 +01:00
}
}
})
}
}
2019-12-22 21:37:26 +01:00
})
2019-12-20 14:52:10 +01:00
}
} else
2019-12-22 21:37:26 +01:00
activity?.let { activity ->
2019-12-27 19:20:38 +01:00
OKDialog.show(activity, resourceHelper.gs(R.string.bolus), resourceHelper.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 14:52:10 +01:00
}
}