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

391 lines
18 KiB
Kotlin
Raw Normal View History

2019-12-20 18:55:54 +01:00
package info.nightscout.androidaps.dialogs
2021-03-25 17:48:07 +01:00
import android.content.Context
import android.os.Bundle
import android.text.Editable
import android.text.TextWatcher
2019-12-20 23:05:35 +01:00
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.view.Window
import android.view.WindowManager
import android.widget.AdapterView
2019-10-10 23:02:35 +02:00
import android.widget.AdapterView.OnItemSelectedListener
import android.widget.ArrayAdapter
import android.widget.CompoundButton
2020-08-21 13:41:48 +02:00
import androidx.fragment.app.FragmentManager
2021-02-06 00:30:27 +01:00
import dagger.android.HasAndroidInjector
2019-12-27 19:20:38 +01:00
import dagger.android.support.DaggerDialogFragment
import info.nightscout.androidaps.Constants
import info.nightscout.androidaps.R
2021-04-29 20:42:45 +02:00
import info.nightscout.androidaps.data.ProfileSealed
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.database.AppRepository
import info.nightscout.androidaps.database.ValueWrapper
2020-12-28 16:42:00 +01:00
import info.nightscout.androidaps.databinding.DialogWizardBinding
2021-03-01 12:42:42 +01:00
import info.nightscout.androidaps.events.EventAutosensCalculationFinished
2019-12-30 13:40:42 +01:00
import info.nightscout.androidaps.logging.AAPSLogger
import info.nightscout.androidaps.logging.LTag
2021-10-15 14:56:22 +02:00
import info.nightscout.androidaps.plugins.bus.RxBus
2019-12-27 19:20:38 +01:00
import info.nightscout.androidaps.plugins.configBuilder.ConstraintChecker
2021-03-25 17:48:07 +01:00
import info.nightscout.androidaps.utils.DateUtil
import info.nightscout.androidaps.utils.DecimalFormatter
import info.nightscout.androidaps.utils.FabricPrivacy
import info.nightscout.androidaps.utils.SafeParse
import info.nightscout.androidaps.utils.ToastUtils
import info.nightscout.androidaps.extensions.toVisibility
import info.nightscout.androidaps.extensions.valueToUnits
2021-04-19 18:44:17 +02:00
import info.nightscout.androidaps.interfaces.*
2019-12-27 19:20:38 +01:00
import info.nightscout.androidaps.utils.resources.ResourceHelper
2021-02-04 20:54:09 +01:00
import info.nightscout.androidaps.utils.rx.AapsSchedulers
2019-12-27 19:20:38 +01:00
import info.nightscout.androidaps.utils.sharedPreferences.SP
2019-12-31 11:57:58 +01:00
import info.nightscout.androidaps.utils.wizard.BolusWizard
import io.reactivex.disposables.CompositeDisposable
import java.text.DecimalFormat
import java.util.*
2019-12-27 19:20:38 +01:00
import javax.inject.Inject
import kotlin.math.abs
2019-12-27 19:20:38 +01:00
class WizardDialog : DaggerDialogFragment() {
2021-02-06 00:30:27 +01:00
@Inject lateinit var injector: HasAndroidInjector
2019-12-30 13:40:42 +01:00
@Inject lateinit var aapsLogger: AAPSLogger
2021-02-04 20:54:09 +01:00
@Inject lateinit var aapsSchedulers: AapsSchedulers
2019-12-30 00:53:44 +01:00
@Inject lateinit var constraintChecker: ConstraintChecker
2021-03-25 17:48:07 +01:00
@Inject lateinit var ctx: Context
2019-12-30 13:40:42 +01:00
@Inject lateinit var sp: SP
2021-10-15 14:56:22 +02:00
@Inject lateinit var rxBus: RxBus
2020-01-03 14:30:39 +01:00
@Inject lateinit var fabricPrivacy: FabricPrivacy
2021-11-04 10:56:12 +01:00
@Inject lateinit var rh: ResourceHelper
2019-12-30 00:53:44 +01:00
@Inject lateinit var profileFunction: ProfileFunction
2021-04-14 00:45:30 +02:00
@Inject lateinit var activePlugin: ActivePlugin
@Inject lateinit var iobCobCalculator: IobCobCalculator
2021-03-01 12:42:42 +01:00
@Inject lateinit var repository: AppRepository
@Inject lateinit var dateUtil: DateUtil
2019-12-27 19:20:38 +01:00
private var wizard: BolusWizard? = null
//one shot guards
private var okClicked: Boolean = false
private val 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) {
calculateInsulin()
}
}
2020-12-28 16:42:00 +01:00
private val timeTextWatcher = 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) {
calculateInsulin()
binding.alarm.isChecked = binding.carbTimeInput.value > 0
}
}
private var disposable: CompositeDisposable = CompositeDisposable()
2020-12-28 16:42:00 +01:00
private var _binding: DialogWizardBinding? = null
// This property is only valid between onCreateView and
// onDestroyView.
private val binding get() = _binding!!
2019-10-10 23:02:35 +02:00
override fun onStart() {
super.onStart()
dialog?.window?.setLayout(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)
}
override fun onSaveInstanceState(savedInstanceState: Bundle) {
super.onSaveInstanceState(savedInstanceState)
2020-12-28 16:42:00 +01:00
savedInstanceState.putDouble("bg_input", binding.bgInput.value)
savedInstanceState.putDouble("carbs_input", binding.carbsInput.value)
savedInstanceState.putDouble("correction_input", binding.correctionInput.value)
savedInstanceState.putDouble("carb_time_input", binding.carbTimeInput.value)
}
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
2020-12-28 16:42:00 +01:00
savedInstanceState: Bundle?): View {
2019-09-15 18:56:54 +02:00
dialog?.window?.requestFeature(Window.FEATURE_NO_TITLE)
dialog?.window?.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN)
isCancelable = true
2019-09-15 18:56:54 +02:00
dialog?.setCanceledOnTouchOutside(false)
2020-12-28 16:42:00 +01:00
_binding = DialogWizardBinding.inflate(inflater, container, false)
return binding.root
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
loadCheckedStates()
processCobCheckBox()
2020-12-28 16:42:00 +01:00
binding.sbcheckbox.visibility = sp.getBoolean(R.string.key_usesuperbolus, false).toVisibility()
binding.notesLayout.visibility = sp.getBoolean(R.string.key_show_notes_entry_dialogs, false).toVisibility()
2019-12-27 19:20:38 +01:00
val maxCarbs = constraintChecker.getMaxCarbsAllowed().value()
val maxCorrection = constraintChecker.getMaxBolusAllowed().value()
2021-04-19 18:44:17 +02:00
if (profileFunction.getUnits() == GlucoseUnit.MGDL)
2020-12-28 16:42:00 +01:00
binding.bgInput.setParams(savedInstanceState?.getDouble("bg_input")
?: 0.0, 0.0, 500.0, 1.0, DecimalFormat("0"), false, binding.ok, timeTextWatcher)
2020-08-19 10:58:45 +02:00
else
2020-12-28 16:42:00 +01:00
binding.bgInput.setParams(savedInstanceState?.getDouble("bg_input")
?: 0.0, 0.0, 30.0, 0.1, DecimalFormat("0.0"), false, binding.ok, textWatcher)
binding.carbsInput.setParams(savedInstanceState?.getDouble("carbs_input")
?: 0.0, 0.0, maxCarbs.toDouble(), 1.0, DecimalFormat("0"), false, binding.ok, textWatcher)
2020-03-16 21:40:29 +01:00
val bolusStep = activePlugin.activePump.pumpDescription.bolusStep
2020-12-28 16:42:00 +01:00
binding.correctionInput.setParams(savedInstanceState?.getDouble("correction_input")
?: 0.0, -maxCorrection, maxCorrection, bolusStep, DecimalFormatter.pumpSupportedBolusFormat(activePlugin.activePump), false, binding.ok, textWatcher)
binding.carbTimeInput.setParams(savedInstanceState?.getDouble("carb_time_input")
?: 0.0, -60.0, 60.0, 5.0, DecimalFormat("0"), false, binding.ok, timeTextWatcher)
initDialog()
2021-11-04 10:56:12 +01:00
binding.percentUsed.text = rh.gs(R.string.format_percent, sp.getInt(R.string.key_boluswizard_percentage, 100))
// ok button
2020-12-28 16:42:00 +01:00
binding.ok.setOnClickListener {
if (okClicked) {
2019-12-30 13:40:42 +01:00
aapsLogger.debug(LTag.UI, "guarding: ok already clicked")
} else {
okClicked = true
calculateInsulin()
2019-12-20 18:55:54 +01:00
context?.let { context ->
wizard?.confirmAndExecute(context)
}
}
dismiss()
}
// cancel button
2020-12-28 16:42:00 +01:00
binding.cancel.setOnClickListener { dismiss() }
// checkboxes
2020-12-28 16:42:00 +01:00
binding.bgcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.ttcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.cobcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.basaliobcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.bolusiobcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.bgtrendcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
binding.sbcheckbox.setOnCheckedChangeListener(::onCheckedChanged)
2019-10-10 23:02:35 +02:00
2020-12-25 12:06:54 +01:00
val showCalc = sp.getBoolean(R.string.key_wizard_calculation_visible, false)
2020-12-28 16:42:00 +01:00
binding.delimiter.visibility = showCalc.toVisibility()
binding.resulttable.visibility = showCalc.toVisibility()
binding.calculationcheckbox.isChecked = showCalc
binding.calculationcheckbox.setOnCheckedChangeListener { _, isChecked ->
2019-10-10 23:02:35 +02:00
run {
2021-11-04 10:56:12 +01:00
sp.putBoolean(rh.gs(R.string.key_wizard_calculation_visible), isChecked)
2020-12-28 16:42:00 +01:00
binding.delimiter.visibility = isChecked.toVisibility()
binding.resulttable.visibility = isChecked.toVisibility()
2019-10-10 23:02:35 +02:00
}
}
// profile spinner
2020-12-28 16:42:00 +01:00
binding.profile.onItemSelectedListener = object : OnItemSelectedListener {
override fun onNothingSelected(parent: AdapterView<*>?) {
2021-11-09 23:06:24 +01:00
ToastUtils.showToastInUiThread(ctx, rh.gs(R.string.noprofileset))
2020-12-28 16:42:00 +01:00
binding.ok.visibility = View.GONE
}
override fun onItemSelected(parent: AdapterView<*>?, view: View?, position: Int, id: Long) {
calculateInsulin()
2020-12-28 16:42:00 +01:00
binding.ok.visibility = View.VISIBLE
}
}
// bus
2019-12-30 00:53:44 +01:00
disposable.add(rxBus
2019-12-20 23:05:35 +01:00
.toObservable(EventAutosensCalculationFinished::class.java)
2021-02-04 20:54:09 +01:00
.observeOn(aapsSchedulers.main)
2019-12-20 23:05:35 +01:00
.subscribe({
activity?.runOnUiThread { calculateInsulin() }
2021-02-04 20:54:09 +01:00
}, fabricPrivacy::logException)
)
}
override fun onDestroyView() {
super.onDestroyView()
disposable.clear()
2020-12-28 16:42:00 +01:00
_binding = null
}
2019-12-21 14:30:14 +01:00
private fun onCheckedChanged(buttonView: CompoundButton, @Suppress("UNUSED_PARAMETER") state: Boolean) {
saveCheckedStates()
2021-04-11 17:58:50 +02:00
binding.ttcheckbox.isEnabled = binding.bgcheckbox.isChecked && repository.getTemporaryTargetActiveAt(dateUtil.now()).blockingGet() is ValueWrapper.Existing
2020-12-28 16:42:00 +01:00
if (buttonView.id == binding.cobcheckbox.id)
processCobCheckBox()
calculateInsulin()
}
private fun processCobCheckBox() {
2020-12-28 16:42:00 +01:00
if (binding.cobcheckbox.isChecked) {
binding.bolusiobcheckbox.isEnabled = false
binding.basaliobcheckbox.isEnabled = false
binding.bolusiobcheckbox.isChecked = true
binding.basaliobcheckbox.isChecked = true
} else {
2020-12-28 16:42:00 +01:00
binding.bolusiobcheckbox.isEnabled = true
binding.basaliobcheckbox.isEnabled = true
}
}
private fun saveCheckedStates() {
2020-12-28 16:42:00 +01:00
sp.putBoolean(R.string.key_wizard_include_cob, binding.cobcheckbox.isChecked)
sp.putBoolean(R.string.key_wizard_include_trend_bg, binding.bgtrendcheckbox.isChecked)
}
private fun loadCheckedStates() {
2020-12-28 16:42:00 +01:00
binding.bgtrendcheckbox.isChecked = sp.getBoolean(R.string.key_wizard_include_trend_bg, false)
binding.cobcheckbox.isChecked = sp.getBoolean(R.string.key_wizard_include_cob, false)
}
private fun valueToUnitsToString(value: Double, units: String): String =
if (units == Constants.MGDL) DecimalFormatter.to0Decimal(value)
else DecimalFormatter.to1Decimal(value * Constants.MGDL_TO_MMOLL)
private fun initDialog() {
2019-12-27 19:20:38 +01:00
val profile = profileFunction.getProfile()
2021-04-14 22:58:21 +02:00
val profileStore = activePlugin.activeProfileSource.profile
if (profile == null || profileStore == null) {
2021-11-04 10:56:12 +01:00
ToastUtils.showToastInUiThread(ctx, rh.gs(R.string.noprofile))
dismiss()
return
}
val profileList: ArrayList<CharSequence> = profileStore.getProfileList()
2021-11-04 10:56:12 +01:00
profileList.add(0, rh.gs(R.string.active))
context?.let { context ->
val adapter = ArrayAdapter(context, R.layout.spinner_centered, profileList)
2020-12-28 16:42:00 +01:00
binding.profile.adapter = adapter
} ?: return
2019-12-27 19:20:38 +01:00
val units = profileFunction.getUnits()
2021-04-19 18:44:17 +02:00
binding.bgunits.text = units.asText
if (units == GlucoseUnit.MGDL)
2020-12-28 16:42:00 +01:00
binding.bgInput.setStep(1.0)
else
2020-12-28 16:42:00 +01:00
binding.bgInput.setStep(0.1)
// Set BG if not old
2021-04-13 18:33:44 +02:00
binding.bgInput.value = iobCobCalculator.ads.actualBg()?.valueToUnits(units) ?: 0.0
2021-04-11 17:58:50 +02:00
binding.ttcheckbox.isEnabled = repository.getTemporaryTargetActiveAt(dateUtil.now()).blockingGet() is ValueWrapper.Existing
// IOB calculation
val bolusIob = iobCobCalculator.calculateIobFromBolus().round()
val basalIob = iobCobCalculator.calculateIobFromTempBasalsIncludingConvertedExtended().round()
2021-11-04 10:56:12 +01:00
binding.bolusiobinsulin.text = rh.gs(R.string.formatinsulinunits, -bolusIob.iob)
binding.basaliobinsulin.text = rh.gs(R.string.formatinsulinunits, -basalIob.basaliob)
calculateInsulin()
2020-12-28 16:42:00 +01:00
binding.percentUsed.visibility = (sp.getInt(R.string.key_boluswizard_percentage, 100) != 100).toVisibility()
}
private fun calculateInsulin() {
2021-04-14 22:58:21 +02:00
val profileStore = activePlugin.activeProfileSource.profile
2020-12-28 16:42:00 +01:00
if (binding.profile.selectedItem == null || profileStore == null)
return // not initialized yet
2020-12-28 16:42:00 +01:00
var profileName = binding.profile.selectedItem.toString()
val specificProfile: Profile?
2021-11-04 10:56:12 +01:00
if (profileName == rh.gs(R.string.active)) {
2019-12-27 19:20:38 +01:00
specificProfile = profileFunction.getProfile()
2019-12-30 13:40:42 +01:00
profileName = profileFunction.getProfileName()
} else
2021-04-29 20:42:45 +02:00
specificProfile = profileStore.getSpecificProfile(profileName)?.let { ProfileSealed.Pure(it) }
if (specificProfile == null) return
// Entered values
2020-12-28 16:42:00 +01:00
var bg = SafeParse.stringToDouble(binding.bgInput.text)
val carbs = SafeParse.stringToInt(binding.carbsInput.text)
val correction = SafeParse.stringToDouble(binding.correctionInput.text)
2019-12-27 19:20:38 +01:00
val carbsAfterConstraint = constraintChecker.applyCarbsConstraints(Constraint(carbs)).value()
if (abs(carbs - carbsAfterConstraint) > 0.01) {
2020-12-28 16:42:00 +01:00
binding.carbsInput.value = 0.0
2021-11-04 10:56:12 +01:00
ToastUtils.showToastInUiThread(ctx, rh.gs(R.string.carbsconstraintapplied))
return
}
2020-12-28 16:42:00 +01:00
bg = if (binding.bgcheckbox.isChecked) bg else 0.0
2021-04-11 17:58:50 +02:00
val dbRecord = repository.getTemporaryTargetActiveAt(dateUtil.now()).blockingGet()
2021-03-07 19:26:08 +01:00
val tempTarget = if (binding.ttcheckbox.isChecked && dbRecord is ValueWrapper.Existing) dbRecord.value else null
// COB
var cob = 0.0
2020-12-28 16:42:00 +01:00
if (binding.cobcheckbox.isChecked) {
val cobInfo = iobCobCalculator.getCobInfo(false, "Wizard COB")
cobInfo.displayCob?.let { cob = it }
}
2020-12-28 16:42:00 +01:00
val carbTime = SafeParse.stringToInt(binding.carbTimeInput.text)
2021-03-25 17:48:07 +01:00
wizard = BolusWizard(injector).doCalc(specificProfile, profileName, tempTarget, carbsAfterConstraint, cob, bg, correction,
sp.getInt(R.string.key_boluswizard_percentage, 100),
2020-12-28 16:42:00 +01:00
binding.bgcheckbox.isChecked,
binding.cobcheckbox.isChecked,
binding.bolusiobcheckbox.isChecked,
binding.basaliobcheckbox.isChecked,
binding.sbcheckbox.isChecked,
binding.ttcheckbox.isChecked,
binding.bgtrendcheckbox.isChecked,
binding.alarm.isChecked,
binding.notes.text.toString(), carbTime)
wizard?.let { wizard ->
2021-11-04 10:56:12 +01:00
binding.bg.text = String.format(rh.gs(R.string.format_bg_isf), valueToUnitsToString(Profile.toMgdl(bg, profileFunction.getUnits()), profileFunction.getUnits().asText), wizard.sens)
binding.bginsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromBG)
2021-11-04 10:56:12 +01:00
binding.carbs.text = String.format(rh.gs(R.string.format_carbs_ic), carbs.toDouble(), wizard.ic)
binding.carbsinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromCarbs)
2021-11-04 10:56:12 +01:00
binding.bolusiobinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromBolusIOB)
binding.basaliobinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromBasalIOB)
2021-11-04 10:56:12 +01:00
binding.correctioninsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromCorrection)
// Superbolus
2021-11-04 10:56:12 +01:00
binding.sb.text = if (binding.sbcheckbox.isChecked) rh.gs(R.string.twohours) else ""
binding.sbinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromSuperBolus)
// Trend
2020-12-28 16:42:00 +01:00
if (binding.bgtrendcheckbox.isChecked && wizard.glucoseStatus != null) {
binding.bgtrend.text = ((if (wizard.trend > 0) "+" else "")
2019-12-27 19:20:38 +01:00
+ Profile.toUnitsString(wizard.trend * 3, wizard.trend * 3 / Constants.MMOLL_TO_MGDL, profileFunction.getUnits())
+ " " + profileFunction.getUnits())
} else {
2020-12-28 16:42:00 +01:00
binding.bgtrend.text = ""
}
2021-11-04 10:56:12 +01:00
binding.bgtrendinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromTrend)
// COB
2020-12-28 16:42:00 +01:00
if (binding.cobcheckbox.isChecked) {
2021-11-04 10:56:12 +01:00
binding.cob.text = String.format(rh.gs(R.string.format_cob_ic), cob, wizard.ic)
binding.cobinsulin.text = rh.gs(R.string.formatinsulinunits, wizard.insulinFromCOB)
} else {
2020-12-28 16:42:00 +01:00
binding.cob.text = ""
binding.cobinsulin.text = ""
}
if (wizard.calculatedTotalInsulin > 0.0 || carbsAfterConstraint > 0.0) {
2021-11-04 10:56:12 +01:00
val insulinText = if (wizard.calculatedTotalInsulin > 0.0) rh.gs(R.string.formatinsulinunits, wizard.calculatedTotalInsulin) else ""
val carbsText = if (carbsAfterConstraint > 0.0) rh.gs(R.string.format_carbs, carbsAfterConstraint) else ""
binding.total.text = rh.gs(R.string.result_insulin_carbs, insulinText, carbsText)
2020-12-28 16:42:00 +01:00
binding.ok.visibility = View.VISIBLE
} else {
2021-11-04 10:56:12 +01:00
binding.total.text = rh.gs(R.string.missing_carbs, wizard.carbsEquivalent.toInt())
2020-12-28 16:42:00 +01:00
binding.ok.visibility = View.INVISIBLE
}
}
}
2020-08-21 13:41:48 +02:00
override fun show(manager: FragmentManager, tag: String?) {
try {
manager.beginTransaction().let {
it.add(this, tag)
it.commitAllowingStateLoss()
}
} catch (e: IllegalStateException) {
2021-08-11 18:10:44 +02:00
aapsLogger.debug(e.localizedMessage ?: "")
2020-08-21 13:41:48 +02:00
}
}
}