package info.nightscout.androidaps.dialogs import android.content.Context 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 android.widget.ArrayAdapter import com.google.common.base.Joiner import info.nightscout.androidaps.Constants import info.nightscout.androidaps.R import info.nightscout.androidaps.data.ProfileSealed import info.nightscout.androidaps.database.AppRepository import info.nightscout.androidaps.database.entities.TemporaryTarget import info.nightscout.androidaps.database.entities.UserEntry.Action import info.nightscout.androidaps.database.entities.UserEntry.Sources import info.nightscout.androidaps.database.entities.ValueWithUnit import info.nightscout.androidaps.database.transactions.InsertAndCancelCurrentTemporaryTargetTransaction import info.nightscout.androidaps.databinding.DialogProfileswitchBinding import info.nightscout.androidaps.extensions.toVisibility import info.nightscout.androidaps.interfaces.ActivePlugin import info.nightscout.androidaps.interfaces.Config import info.nightscout.androidaps.interfaces.Profile import info.nightscout.androidaps.interfaces.ProfileFunction import info.nightscout.shared.logging.LTag import info.nightscout.androidaps.logging.UserEntryLogger import info.nightscout.androidaps.plugins.bus.RxBus import info.nightscout.androidaps.utils.DefaultValueHelper import info.nightscout.androidaps.utils.HardLimits import info.nightscout.androidaps.utils.HtmlHelper import info.nightscout.androidaps.utils.T import info.nightscout.androidaps.utils.ToastUtils import info.nightscout.androidaps.utils.alertDialogs.OKDialog import info.nightscout.androidaps.utils.protection.ProtectionCheck import info.nightscout.androidaps.utils.protection.ProtectionCheck.Protection.BOLUS import info.nightscout.androidaps.interfaces.ResourceHelper import io.reactivex.rxjava3.disposables.CompositeDisposable import io.reactivex.rxjava3.kotlin.plusAssign import java.text.DecimalFormat import java.util.* import java.util.concurrent.TimeUnit import javax.inject.Inject import kotlin.collections.ArrayList class ProfileSwitchDialog : DialogFragmentWithDate() { @Inject lateinit var rh: ResourceHelper @Inject lateinit var profileFunction: ProfileFunction @Inject lateinit var activePlugin: ActivePlugin @Inject lateinit var repository: AppRepository @Inject lateinit var uel: UserEntryLogger @Inject lateinit var config: Config @Inject lateinit var hardLimits: HardLimits @Inject lateinit var rxBus: RxBus @Inject lateinit var defaultValueHelper: DefaultValueHelper @Inject lateinit var ctx: Context @Inject lateinit var protectionCheck: ProtectionCheck private var queryingProtection = false private var profileName: String? = null private val disposable = CompositeDisposable() private var _binding: DialogProfileswitchBinding? = null // This property is only valid between onCreateView and onDestroyView. private val binding get() = _binding!! private val textWatcher: TextWatcher = object : TextWatcher { override fun afterTextChanged(s: Editable) { val isDuration = binding.duration.value > 0 val isLowerPercentage = binding.percentage.value < 100 binding.ttLayout.visibility = (isDuration && isLowerPercentage).toVisibility() } override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {} override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {} } override fun onSaveInstanceState(savedInstanceState: Bundle) { super.onSaveInstanceState(savedInstanceState) savedInstanceState.putDouble("duration", binding.duration.value) savedInstanceState.putDouble("percentage", binding.percentage.value) savedInstanceState.putDouble("timeshift", binding.timeshift.value) } override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View { onCreateViewGeneral() arguments?.let { bundle -> profileName = bundle.getString("profileName", null) } _binding = DialogProfileswitchBinding.inflate(inflater, container, false) return binding.root } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) binding.duration.setParams( savedInstanceState?.getDouble("duration") ?: 0.0, 0.0, Constants.MAX_PROFILE_SWITCH_DURATION, 10.0, DecimalFormat("0"), false, binding.okcancel.ok, textWatcher ) binding.percentage.setParams( savedInstanceState?.getDouble("percentage") ?: 100.0, Constants.CPP_MIN_PERCENTAGE.toDouble(), Constants.CPP_MAX_PERCENTAGE.toDouble(), 5.0, DecimalFormat("0"), false, binding.okcancel.ok, textWatcher ) binding.timeshift.setParams( savedInstanceState?.getDouble("timeshift") ?: 0.0, Constants.CPP_MIN_TIMESHIFT.toDouble(), Constants.CPP_MAX_TIMESHIFT.toDouble(), 1.0, DecimalFormat("0"), false, binding.okcancel.ok ) // profile context?.let { context -> val profileStore = activePlugin.activeProfileSource.profile ?: return val profileListToCheck = profileStore.getProfileList() val profileList = ArrayList() for (profileName in profileListToCheck) { val profileToCheck = activePlugin.activeProfileSource.profile?.getSpecificProfile(profileName.toString()) if (profileToCheck != null && ProfileSealed.Pure(profileToCheck).isValid("ProfileSwitch", activePlugin.activePump, config, rh, rxBus, hardLimits, false).isValid) profileList.add(profileName) } if (profileList.isEmpty()) { dismiss() return } binding.profileList.setAdapter(ArrayAdapter(context, R.layout.spinner_centered, profileList)) // set selected to actual profile if (profileName != null) binding.profileList.setText(profileName, false) else { binding.profileList.setText(profileList[0], false) for (p in profileList.indices) if (profileList[p] == profileFunction.getOriginalProfileName()) binding.profileList.setText(profileList[p], false) } } profileFunction.getProfile()?.let { profile -> if (profile is ProfileSealed.EPS) if (profile.value.originalPercentage != 100 || profile.value.originalTimeshift != 0L) { binding.reuselayout.visibility = View.VISIBLE binding.reusebutton.text = rh.gs(R.string.reuse_profile_pct_hours, profile.value.originalPercentage, T.msecs(profile.value.originalTimeshift).hours().toInt()) binding.reusebutton.setOnClickListener { binding.percentage.value = profile.value.originalPercentage.toDouble() binding.timeshift.value = T.msecs(profile.value.originalTimeshift).hours().toDouble() } } } binding.ttLayout.visibility = View.GONE binding.durationLabel.labelFor = binding.duration.editTextId binding.percentageLabel.labelFor = binding.percentage.editTextId binding.timeshiftLabel.labelFor = binding.timeshift.editTextId } override fun onDestroyView() { super.onDestroyView() disposable.clear() _binding = null } override fun submit(): Boolean { if (_binding == null) return false val profileStore = activePlugin.activeProfileSource.profile ?: return false val actions: LinkedList = LinkedList() val duration = binding.duration.value.toInt() if (duration > 0L) actions.add(rh.gs(R.string.duration) + ": " + rh.gs(R.string.format_mins, duration)) val profileName = binding.profileList.text.toString() actions.add(rh.gs(R.string.profile) + ": " + profileName) val percent = binding.percentage.value.toInt() if (percent != 100) actions.add(rh.gs(R.string.percent) + ": " + percent + "%") val timeShift = binding.timeshift.value.toInt() if (timeShift != 0) actions.add(rh.gs(R.string.careportal_newnstreatment_timeshift_label) + ": " + rh.gs(R.string.format_hours, timeShift.toDouble())) val notes = binding.notesLayout.notes.text.toString() if (notes.isNotEmpty()) actions.add(rh.gs(R.string.notes_label) + ": " + notes) if (eventTimeChanged) actions.add(rh.gs(R.string.time) + ": " + dateUtil.dateAndTimeString(eventTime)) val isTT = binding.duration.value > 0 && binding.percentage.value < 100 && binding.tt.isChecked val target = defaultValueHelper.determineActivityTT() val units = profileFunction.getUnits() if (isTT) actions.add(rh.gs(R.string.careportal_temporarytarget) + ": " + rh.gs(R.string.activity)) activity?.let { activity -> val ps = profileFunction.buildProfileSwitch(profileStore, profileName, duration, percent, timeShift, eventTime) ?: return@let val validity = ProfileSealed.PS(ps).isValid(rh.gs(R.string.careportal_profileswitch), activePlugin.activePump, config, rh, rxBus, hardLimits, false) if (validity.isValid) OKDialog.showConfirmation(activity, rh.gs(R.string.careportal_profileswitch), HtmlHelper.fromHtml(Joiner.on("
").join(actions)), { if (profileFunction.createProfileSwitch( profileStore, profileName = profileName, durationInMinutes = duration, percentage = percent, timeShiftInHours = timeShift, timestamp = eventTime ) ) { uel.log(Action.PROFILE_SWITCH, Sources.ProfileSwitchDialog, notes, ValueWithUnit.Timestamp(eventTime).takeIf { eventTimeChanged }, ValueWithUnit.SimpleString(profileName), ValueWithUnit.Percent(percent), ValueWithUnit.Hour(timeShift).takeIf { timeShift != 0 }, ValueWithUnit.Minute(duration).takeIf { duration != 0 }) if (percent == 90 && duration == 10) sp.putBoolean(R.string.key_objectiveuseprofileswitch, true) if (isTT) { disposable += repository.runTransactionForResult( InsertAndCancelCurrentTemporaryTargetTransaction( timestamp = eventTime, duration = TimeUnit.MINUTES.toMillis(duration.toLong()), reason = TemporaryTarget.Reason.ACTIVITY, lowTarget = Profile.toMgdl(target, profileFunction.getUnits()), highTarget = Profile.toMgdl(target, profileFunction.getUnits()) ) ).subscribe({ result -> result.inserted.forEach { aapsLogger.debug(LTag.DATABASE, "Inserted temp target $it") } result.updated.forEach { aapsLogger.debug(LTag.DATABASE, "Updated temp target $it") } }, { aapsLogger.error(LTag.DATABASE, "Error while saving temporary target", it) }) uel.log( Action.TT, Sources.TTDialog, ValueWithUnit.Timestamp(eventTime).takeIf { eventTimeChanged }, ValueWithUnit.TherapyEventTTReason( TemporaryTarget.Reason.ACTIVITY ), ValueWithUnit.fromGlucoseUnit(target, units.asText), ValueWithUnit.Minute(duration) ) } } }) else { OKDialog.show( activity, rh.gs(R.string.careportal_profileswitch), HtmlHelper.fromHtml(Joiner.on("
").join(validity.reasons)) ) return false } } return true } override fun onResume() { super.onResume() 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) } } } }