TimeListEdit -> kt
This commit is contained in:
parent
3bc62e9a01
commit
971bf87cac
8 changed files with 482 additions and 526 deletions
|
@ -1,34 +1,31 @@
|
||||||
package info.nightscout.interfaces.profile
|
package info.nightscout.interfaces.profile
|
||||||
|
|
||||||
import androidx.fragment.app.FragmentActivity
|
import androidx.fragment.app.FragmentActivity
|
||||||
import info.nightscout.interfaces.Constants
|
|
||||||
import org.json.JSONArray
|
import org.json.JSONArray
|
||||||
|
|
||||||
interface ProfileSource {
|
interface ProfileSource {
|
||||||
|
|
||||||
class SingleProfile {
|
class SingleProfile(
|
||||||
|
var name: String,
|
||||||
var name: String? = null
|
var mgdl: Boolean,
|
||||||
var mgdl: Boolean = false
|
var dia: Double,
|
||||||
var dia: Double = Constants.defaultDIA
|
var ic: JSONArray,
|
||||||
var ic: JSONArray? = null
|
var isf: JSONArray,
|
||||||
var isf: JSONArray? = null
|
var basal: JSONArray,
|
||||||
var basal: JSONArray? = null
|
var targetLow: JSONArray,
|
||||||
var targetLow: JSONArray? = null
|
var targetHigh: JSONArray,
|
||||||
var targetHigh: JSONArray? = null
|
) {
|
||||||
|
fun deepClone(): SingleProfile =
|
||||||
fun deepClone(): SingleProfile {
|
SingleProfile(
|
||||||
val sp = SingleProfile()
|
name = name,
|
||||||
sp.name = name
|
mgdl = mgdl,
|
||||||
sp.mgdl = mgdl
|
dia = dia,
|
||||||
sp.dia = dia
|
ic = JSONArray(ic.toString()),
|
||||||
sp.ic = JSONArray(ic.toString())
|
isf = JSONArray(isf.toString()),
|
||||||
sp.isf = JSONArray(isf.toString())
|
basal = JSONArray(basal.toString()),
|
||||||
sp.basal = JSONArray(basal.toString())
|
targetLow = JSONArray(targetLow.toString()),
|
||||||
sp.targetLow = JSONArray(targetLow.toString())
|
targetHigh = JSONArray(targetHigh.toString())
|
||||||
sp.targetHigh = JSONArray(targetHigh.toString())
|
)
|
||||||
return sp
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
val profile: ProfileStore?
|
val profile: ProfileStore?
|
||||||
|
|
|
@ -33,8 +33,8 @@ import info.nightscout.interfaces.queue.CommandQueue
|
||||||
import info.nightscout.interfaces.ui.UiInteraction
|
import info.nightscout.interfaces.ui.UiInteraction
|
||||||
import info.nightscout.plugins.R
|
import info.nightscout.plugins.R
|
||||||
import info.nightscout.plugins.databinding.ActionsFragmentBinding
|
import info.nightscout.plugins.databinding.ActionsFragmentBinding
|
||||||
|
import info.nightscout.plugins.general.overview.ui.StatusLightHandler
|
||||||
import info.nightscout.plugins.skins.SkinProvider
|
import info.nightscout.plugins.skins.SkinProvider
|
||||||
import info.nightscout.plugins.ui.StatusLightHandler
|
|
||||||
import info.nightscout.rx.AapsSchedulers
|
import info.nightscout.rx.AapsSchedulers
|
||||||
import info.nightscout.rx.bus.RxBus
|
import info.nightscout.rx.bus.RxBus
|
||||||
import info.nightscout.rx.events.EventCustomActionsChanged
|
import info.nightscout.rx.events.EventCustomActionsChanged
|
||||||
|
|
|
@ -74,8 +74,8 @@ import info.nightscout.plugins.databinding.OverviewFragmentBinding
|
||||||
import info.nightscout.plugins.general.overview.graphData.GraphData
|
import info.nightscout.plugins.general.overview.graphData.GraphData
|
||||||
import info.nightscout.plugins.general.overview.notifications.NotificationStore
|
import info.nightscout.plugins.general.overview.notifications.NotificationStore
|
||||||
import info.nightscout.plugins.general.overview.notifications.events.EventUpdateOverviewNotification
|
import info.nightscout.plugins.general.overview.notifications.events.EventUpdateOverviewNotification
|
||||||
|
import info.nightscout.plugins.general.overview.ui.StatusLightHandler
|
||||||
import info.nightscout.plugins.skins.SkinProvider
|
import info.nightscout.plugins.skins.SkinProvider
|
||||||
import info.nightscout.plugins.ui.StatusLightHandler
|
|
||||||
import info.nightscout.rx.AapsSchedulers
|
import info.nightscout.rx.AapsSchedulers
|
||||||
import info.nightscout.rx.bus.RxBus
|
import info.nightscout.rx.bus.RxBus
|
||||||
import info.nightscout.rx.events.EventAcceptOpenLoopChange
|
import info.nightscout.rx.events.EventAcceptOpenLoopChange
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package info.nightscout.plugins.ui
|
package info.nightscout.plugins.general.overview.ui
|
||||||
|
|
||||||
import android.annotation.SuppressLint
|
import android.annotation.SuppressLint
|
||||||
import android.os.Handler
|
import android.os.Handler
|
|
@ -27,7 +27,7 @@ import info.nightscout.interfaces.utils.DecimalFormatter
|
||||||
import info.nightscout.interfaces.utils.HardLimits
|
import info.nightscout.interfaces.utils.HardLimits
|
||||||
import info.nightscout.plugins.R
|
import info.nightscout.plugins.R
|
||||||
import info.nightscout.plugins.databinding.ProfileFragmentBinding
|
import info.nightscout.plugins.databinding.ProfileFragmentBinding
|
||||||
import info.nightscout.plugins.ui.TimeListEdit
|
import info.nightscout.plugins.profile.ui.TimeListEdit
|
||||||
import info.nightscout.rx.AapsSchedulers
|
import info.nightscout.rx.AapsSchedulers
|
||||||
import info.nightscout.rx.bus.RxBus
|
import info.nightscout.rx.bus.RxBus
|
||||||
import info.nightscout.rx.events.EventLocalProfileChanged
|
import info.nightscout.rx.events.EventLocalProfileChanged
|
||||||
|
@ -139,10 +139,10 @@ class ProfileFragment : DaggerFragment() {
|
||||||
binding.dia.setParams(currentProfile.dia, hardLimits.minDia(), hardLimits.maxDia(), 0.1, DecimalFormat("0.0"), false, null, textWatch)
|
binding.dia.setParams(currentProfile.dia, hardLimits.minDia(), hardLimits.maxDia(), 0.1, DecimalFormat("0.0"), false, null, textWatch)
|
||||||
binding.dia.tag = "LP_DIA"
|
binding.dia.tag = "LP_DIA"
|
||||||
TimeListEdit(
|
TimeListEdit(
|
||||||
context,
|
requireContext(),
|
||||||
aapsLogger,
|
aapsLogger,
|
||||||
dateUtil,
|
dateUtil,
|
||||||
view,
|
requireView(),
|
||||||
R.id.ic_holder,
|
R.id.ic_holder,
|
||||||
"IC",
|
"IC",
|
||||||
rh.gs(info.nightscout.core.ui.R.string.ic_long_label),
|
rh.gs(info.nightscout.core.ui.R.string.ic_long_label),
|
||||||
|
@ -156,10 +156,10 @@ class ProfileFragment : DaggerFragment() {
|
||||||
)
|
)
|
||||||
basalView =
|
basalView =
|
||||||
TimeListEdit(
|
TimeListEdit(
|
||||||
context,
|
requireContext(),
|
||||||
aapsLogger,
|
aapsLogger,
|
||||||
dateUtil,
|
dateUtil,
|
||||||
view,
|
requireView(),
|
||||||
R.id.basal_holder,
|
R.id.basal_holder,
|
||||||
"BASAL",
|
"BASAL",
|
||||||
rh.gs(info.nightscout.core.ui.R.string.basal_long_label) + ": " + sumLabel(),
|
rh.gs(info.nightscout.core.ui.R.string.basal_long_label) + ": " + sumLabel(),
|
||||||
|
@ -173,12 +173,27 @@ class ProfileFragment : DaggerFragment() {
|
||||||
)
|
)
|
||||||
if (units == Constants.MGDL) {
|
if (units == Constants.MGDL) {
|
||||||
val isfRange = doubleArrayOf(HardLimits.MIN_ISF, HardLimits.MAX_ISF)
|
val isfRange = doubleArrayOf(HardLimits.MIN_ISF, HardLimits.MAX_ISF)
|
||||||
TimeListEdit(context, aapsLogger, dateUtil, view, R.id.isf_holder, "ISF", rh.gs(info.nightscout.core.ui.R.string.isf_long_label), currentProfile.isf, null, isfRange, null, 1.0, DecimalFormat("0"), save)
|
|
||||||
TimeListEdit(
|
TimeListEdit(
|
||||||
context,
|
requireContext(),
|
||||||
aapsLogger,
|
aapsLogger,
|
||||||
dateUtil,
|
dateUtil,
|
||||||
view,
|
requireView(),
|
||||||
|
R.id.isf_holder,
|
||||||
|
"ISF",
|
||||||
|
rh.gs(info.nightscout.core.ui.R.string.isf_long_label),
|
||||||
|
currentProfile.isf,
|
||||||
|
null,
|
||||||
|
isfRange,
|
||||||
|
null,
|
||||||
|
1.0,
|
||||||
|
DecimalFormat("0"),
|
||||||
|
save
|
||||||
|
)
|
||||||
|
TimeListEdit(
|
||||||
|
requireContext(),
|
||||||
|
aapsLogger,
|
||||||
|
dateUtil,
|
||||||
|
requireView(),
|
||||||
R.id.target_holder,
|
R.id.target_holder,
|
||||||
"TARGET",
|
"TARGET",
|
||||||
rh.gs(info.nightscout.core.ui.R.string.target_long_label),
|
rh.gs(info.nightscout.core.ui.R.string.target_long_label),
|
||||||
|
@ -195,7 +210,9 @@ class ProfileFragment : DaggerFragment() {
|
||||||
roundUp(Profile.fromMgdlToUnits(HardLimits.MIN_ISF, GlucoseUnit.MMOL)),
|
roundUp(Profile.fromMgdlToUnits(HardLimits.MIN_ISF, GlucoseUnit.MMOL)),
|
||||||
roundDown(Profile.fromMgdlToUnits(HardLimits.MAX_ISF, GlucoseUnit.MMOL))
|
roundDown(Profile.fromMgdlToUnits(HardLimits.MAX_ISF, GlucoseUnit.MMOL))
|
||||||
)
|
)
|
||||||
TimeListEdit(context, aapsLogger, dateUtil, view, R.id.isf_holder, "ISF", rh.gs(info.nightscout.core.ui.R.string.isf_long_label), currentProfile.isf, null, isfRange, null, 0.1, DecimalFormat("0.0"), save)
|
TimeListEdit(requireContext(), aapsLogger, dateUtil, requireView(), R.id.isf_holder, "ISF", rh.gs(info.nightscout.core.ui.R.string.isf_long_label), currentProfile.isf, null, isfRange, null, 0.1,
|
||||||
|
DecimalFormat
|
||||||
|
("0.0"), save)
|
||||||
val range1 = doubleArrayOf(
|
val range1 = doubleArrayOf(
|
||||||
roundUp(Profile.fromMgdlToUnits(HardLimits.VERY_HARD_LIMIT_MIN_BG[0], GlucoseUnit.MMOL)),
|
roundUp(Profile.fromMgdlToUnits(HardLimits.VERY_HARD_LIMIT_MIN_BG[0], GlucoseUnit.MMOL)),
|
||||||
roundDown(Profile.fromMgdlToUnits(HardLimits.VERY_HARD_LIMIT_MIN_BG[1], GlucoseUnit.MMOL))
|
roundDown(Profile.fromMgdlToUnits(HardLimits.VERY_HARD_LIMIT_MIN_BG[1], GlucoseUnit.MMOL))
|
||||||
|
@ -206,10 +223,10 @@ class ProfileFragment : DaggerFragment() {
|
||||||
)
|
)
|
||||||
aapsLogger.info(LTag.CORE, "TimeListEdit", "build: range1" + range1[0] + " " + range1[1] + " range2" + range2[0] + " " + range2[1])
|
aapsLogger.info(LTag.CORE, "TimeListEdit", "build: range1" + range1[0] + " " + range1[1] + " range2" + range2[0] + " " + range2[1])
|
||||||
TimeListEdit(
|
TimeListEdit(
|
||||||
context,
|
requireContext(),
|
||||||
aapsLogger,
|
aapsLogger,
|
||||||
dateUtil,
|
dateUtil,
|
||||||
view,
|
requireView(),
|
||||||
R.id.target_holder,
|
R.id.target_holder,
|
||||||
"TARGET",
|
"TARGET",
|
||||||
rh.gs(info.nightscout.core.ui.R.string.target_long_label),
|
rh.gs(info.nightscout.core.ui.R.string.target_long_label),
|
||||||
|
|
|
@ -99,7 +99,7 @@ class ProfilePlugin @Inject constructor(
|
||||||
ToastUtils.errorToast(activity, rh.gs(info.nightscout.core.ui.R.string.value_out_of_hard_limits, rh.gs(info.nightscout.core.ui.R.string.profile_dia), dia))
|
ToastUtils.errorToast(activity, rh.gs(info.nightscout.core.ui.R.string.value_out_of_hard_limits, rh.gs(info.nightscout.core.ui.R.string.profile_dia), dia))
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if (name.isNullOrEmpty()) {
|
if (name.isEmpty()) {
|
||||||
ToastUtils.errorToast(activity, rh.gs(R.string.missing_profile_name))
|
ToastUtils.errorToast(activity, rh.gs(R.string.missing_profile_name))
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -177,7 +177,6 @@ class ProfilePlugin @Inject constructor(
|
||||||
override fun storeSettings(activity: FragmentActivity?) {
|
override fun storeSettings(activity: FragmentActivity?) {
|
||||||
for (i in 0 until numOfProfiles) {
|
for (i in 0 until numOfProfiles) {
|
||||||
profiles[i].run {
|
profiles[i].run {
|
||||||
name?.let { name ->
|
|
||||||
val localProfileNumbered = Constants.LOCAL_PROFILE + "_" + i + "_"
|
val localProfileNumbered = Constants.LOCAL_PROFILE + "_" + i + "_"
|
||||||
sp.putString(localProfileNumbered + "name", name)
|
sp.putString(localProfileNumbered + "name", name)
|
||||||
sp.putBoolean(localProfileNumbered + "mgdl", mgdl)
|
sp.putBoolean(localProfileNumbered + "mgdl", mgdl)
|
||||||
|
@ -189,7 +188,6 @@ class ProfilePlugin @Inject constructor(
|
||||||
sp.putString(localProfileNumbered + "targethigh", targetHigh.toString())
|
sp.putString(localProfileNumbered + "targethigh", targetHigh.toString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
sp.putInt(Constants.LOCAL_PROFILE + "_profiles", numOfProfiles)
|
sp.putInt(Constants.LOCAL_PROFILE + "_profiles", numOfProfiles)
|
||||||
|
|
||||||
sp.putLong(info.nightscout.core.utils.R.string.key_local_profile_last_change, dateUtil.now())
|
sp.putLong(info.nightscout.core.utils.R.string.key_local_profile_last_change, dateUtil.now())
|
||||||
|
@ -198,10 +196,7 @@ class ProfilePlugin @Inject constructor(
|
||||||
aapsLogger.debug(LTag.PROFILE, "Storing settings: " + rawProfile?.data.toString())
|
aapsLogger.debug(LTag.PROFILE, "Storing settings: " + rawProfile?.data.toString())
|
||||||
rxBus.send(EventProfileStoreChanged())
|
rxBus.send(EventProfileStoreChanged())
|
||||||
var namesOK = true
|
var namesOK = true
|
||||||
profiles.forEach {
|
profiles.forEach { if (it.name.contains(".")) namesOK = false }
|
||||||
val name = it.name ?: "."
|
|
||||||
if (name.contains(".")) namesOK = false
|
|
||||||
}
|
|
||||||
if (!namesOK) activity?.let {
|
if (!namesOK) activity?.let {
|
||||||
OKDialog.show(it, "", rh.gs(R.string.profile_name_contains_dot))
|
OKDialog.show(it, "", rh.gs(R.string.profile_name_contains_dot))
|
||||||
}
|
}
|
||||||
|
@ -214,20 +209,22 @@ class ProfilePlugin @Inject constructor(
|
||||||
// numOfProfiles = max(numOfProfiles, 1) // create at least one default profile if none exists
|
// numOfProfiles = max(numOfProfiles, 1) // create at least one default profile if none exists
|
||||||
|
|
||||||
for (i in 0 until numOfProfiles) {
|
for (i in 0 until numOfProfiles) {
|
||||||
val p = ProfileSource.SingleProfile()
|
|
||||||
val localProfileNumbered = Constants.LOCAL_PROFILE + "_" + i + "_"
|
val localProfileNumbered = Constants.LOCAL_PROFILE + "_" + i + "_"
|
||||||
|
val name = sp.getString(localProfileNumbered + "name", Constants.LOCAL_PROFILE + i)
|
||||||
p.name = sp.getString(localProfileNumbered + "name", Constants.LOCAL_PROFILE + i)
|
if (isExistingName(name)) continue
|
||||||
if (isExistingName(p.name)) continue
|
|
||||||
p.mgdl = sp.getBoolean(localProfileNumbered + "mgdl", false)
|
|
||||||
p.dia = sp.getDouble(localProfileNumbered + "dia", Constants.defaultDIA)
|
|
||||||
try {
|
try {
|
||||||
p.ic = JSONArray(sp.getString(localProfileNumbered + "ic", defaultArray))
|
profiles.add(
|
||||||
p.isf = JSONArray(sp.getString(localProfileNumbered + "isf", defaultArray))
|
ProfileSource.SingleProfile(
|
||||||
p.basal = JSONArray(sp.getString(localProfileNumbered + "basal", defaultArray))
|
name = name,
|
||||||
p.targetLow = JSONArray(sp.getString(localProfileNumbered + "targetlow", defaultArray))
|
mgdl = sp.getBoolean(localProfileNumbered + "mgdl", false),
|
||||||
p.targetHigh = JSONArray(sp.getString(localProfileNumbered + "targethigh", defaultArray))
|
dia = sp.getDouble(localProfileNumbered + "dia", Constants.defaultDIA),
|
||||||
profiles.add(p)
|
ic = JSONArray(sp.getString(localProfileNumbered + "ic", defaultArray)),
|
||||||
|
isf = JSONArray(sp.getString(localProfileNumbered + "isf", defaultArray)),
|
||||||
|
basal = JSONArray(sp.getString(localProfileNumbered + "basal", defaultArray)),
|
||||||
|
targetLow = JSONArray(sp.getString(localProfileNumbered + "targetlow", defaultArray)),
|
||||||
|
targetHigh = JSONArray(sp.getString(localProfileNumbered + "targethigh", defaultArray))
|
||||||
|
)
|
||||||
|
)
|
||||||
} catch (e: JSONException) {
|
} catch (e: JSONException) {
|
||||||
aapsLogger.error("Exception", e)
|
aapsLogger.error("Exception", e)
|
||||||
}
|
}
|
||||||
|
@ -279,16 +276,16 @@ class ProfilePlugin @Inject constructor(
|
||||||
}
|
}
|
||||||
val profile = ProfileSealed.Pure(pureProfile)
|
val profile = ProfileSealed.Pure(pureProfile)
|
||||||
val pureJson = pureProfile.jsonObject
|
val pureJson = pureProfile.jsonObject
|
||||||
val sp = ProfileSource.SingleProfile()
|
return ProfileSource.SingleProfile(
|
||||||
sp.name = verifiedName
|
name = verifiedName,
|
||||||
sp.mgdl = profile.units == GlucoseUnit.MGDL
|
mgdl = profile.units == GlucoseUnit.MGDL,
|
||||||
sp.dia = pureJson.getDouble("dia")
|
dia = pureJson.getDouble("dia"),
|
||||||
sp.ic = pureJson.getJSONArray("carbratio")
|
ic = pureJson.getJSONArray("carbratio"),
|
||||||
sp.isf = pureJson.getJSONArray("sens")
|
isf = pureJson.getJSONArray("sens"),
|
||||||
sp.basal = pureJson.getJSONArray("basal")
|
basal = pureJson.getJSONArray("basal"),
|
||||||
sp.targetLow = pureJson.getJSONArray("target_low")
|
targetLow = pureJson.getJSONArray("target_low"),
|
||||||
sp.targetHigh = pureJson.getJSONArray("target_high")
|
targetHigh = pureJson.getJSONArray("target_high")
|
||||||
return sp
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun isExistingName(name: String?): Boolean {
|
private fun isExistingName(name: String?): Boolean {
|
||||||
|
@ -348,16 +345,18 @@ class ProfilePlugin @Inject constructor(
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
val p = ProfileSource.SingleProfile()
|
profiles.add(
|
||||||
p.name = Constants.LOCAL_PROFILE + free
|
ProfileSource.SingleProfile(
|
||||||
p.mgdl = profileFunction.getUnits() == GlucoseUnit.MGDL
|
name = Constants.LOCAL_PROFILE + free,
|
||||||
p.dia = Constants.defaultDIA
|
mgdl = profileFunction.getUnits() == GlucoseUnit.MGDL,
|
||||||
p.ic = JSONArray(defaultArray)
|
dia = Constants.defaultDIA,
|
||||||
p.isf = JSONArray(defaultArray)
|
ic = JSONArray(defaultArray),
|
||||||
p.basal = JSONArray(defaultArray)
|
isf = JSONArray(defaultArray),
|
||||||
p.targetLow = JSONArray(defaultArray)
|
basal = JSONArray(defaultArray),
|
||||||
p.targetHigh = JSONArray(defaultArray)
|
targetLow = JSONArray(defaultArray),
|
||||||
profiles.add(p)
|
targetHigh = JSONArray(defaultArray)
|
||||||
|
)
|
||||||
|
)
|
||||||
currentProfileIndex = profiles.size - 1
|
currentProfileIndex = profiles.size - 1
|
||||||
createAndStoreConvertedProfile()
|
createAndStoreConvertedProfile()
|
||||||
storeSettings()
|
storeSettings()
|
||||||
|
@ -397,7 +396,6 @@ class ProfilePlugin @Inject constructor(
|
||||||
try {
|
try {
|
||||||
for (i in 0 until numOfProfiles) {
|
for (i in 0 until numOfProfiles) {
|
||||||
profiles[i].run {
|
profiles[i].run {
|
||||||
name?.let { name ->
|
|
||||||
val profile = JSONObject()
|
val profile = JSONObject()
|
||||||
profile.put("dia", dia)
|
profile.put("dia", dia)
|
||||||
profile.put("carbratio", ic)
|
profile.put("carbratio", ic)
|
||||||
|
@ -410,7 +408,6 @@ class ProfilePlugin @Inject constructor(
|
||||||
store.put(name, profile)
|
store.put(name, profile)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if (numOfProfiles > 0) json.put("defaultProfile", currentProfile()?.name)
|
if (numOfProfiles > 0) json.put("defaultProfile", currentProfile()?.name)
|
||||||
val startDate = sp.getLong(info.nightscout.core.utils.R.string.key_local_profile_last_change, dateUtil.now())
|
val startDate = sp.getLong(info.nightscout.core.utils.R.string.key_local_profile_last_change, dateUtil.now())
|
||||||
json.put("date", startDate)
|
json.put("date", startDate)
|
||||||
|
|
|
@ -0,0 +1,382 @@
|
||||||
|
package info.nightscout.plugins.profile.ui
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.text.Editable
|
||||||
|
import android.text.TextWatcher
|
||||||
|
import android.view.Gravity
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.widget.AdapterView
|
||||||
|
import android.widget.ArrayAdapter
|
||||||
|
import android.widget.ImageView
|
||||||
|
import android.widget.LinearLayout
|
||||||
|
import android.widget.TextView
|
||||||
|
import androidx.core.widget.TextViewCompat
|
||||||
|
import info.nightscout.core.ui.elements.NumberPicker
|
||||||
|
import info.nightscout.core.ui.elements.SpinnerHelper
|
||||||
|
import info.nightscout.rx.logging.AAPSLogger
|
||||||
|
import info.nightscout.shared.SafeParse.stringToDouble
|
||||||
|
import info.nightscout.shared.utils.DateUtil
|
||||||
|
import org.json.JSONArray
|
||||||
|
import org.json.JSONException
|
||||||
|
import org.json.JSONObject
|
||||||
|
import java.text.DecimalFormat
|
||||||
|
import java.text.NumberFormat
|
||||||
|
|
||||||
|
class TimeListEdit(
|
||||||
|
private val context: Context,
|
||||||
|
private val aapsLogger: AAPSLogger,
|
||||||
|
private val dateUtil: DateUtil,
|
||||||
|
private val view: View,
|
||||||
|
private val resLayoutId: Int,
|
||||||
|
private val tagPrefix: String,
|
||||||
|
private var label: String,
|
||||||
|
private val data1: JSONArray,
|
||||||
|
private val data2: JSONArray?,
|
||||||
|
range1: DoubleArray,
|
||||||
|
range2: DoubleArray?,
|
||||||
|
private val step: Double,
|
||||||
|
formatter: NumberFormat,
|
||||||
|
save: Runnable?
|
||||||
|
) {
|
||||||
|
|
||||||
|
private val intervals = arrayOfNulls<View>(24)
|
||||||
|
private val spinners = arrayOfNulls<SpinnerHelper>(24)
|
||||||
|
private val numberPickers1 = arrayOfNulls<NumberPicker>(24)
|
||||||
|
private val numberPickers2 = arrayOfNulls<NumberPicker>(24)
|
||||||
|
private val addButtons = arrayOfNulls<ImageView>(24)
|
||||||
|
private val removeButtons = arrayOfNulls<ImageView>(24)
|
||||||
|
private var finalAdd: ImageView? = null
|
||||||
|
private val min: Double
|
||||||
|
private val max: Double
|
||||||
|
private val min2: Double
|
||||||
|
private val max2: Double
|
||||||
|
private val formatter: NumberFormat
|
||||||
|
private val save: Runnable?
|
||||||
|
private var layout: LinearLayout? = null
|
||||||
|
private var textLabel: TextView? = null
|
||||||
|
private var inflatedUntil = -1
|
||||||
|
|
||||||
|
init {
|
||||||
|
min = range1[0]
|
||||||
|
max = range1[1]
|
||||||
|
min2 = range2?.get(0) ?: 0.0
|
||||||
|
max2 = range2?.get(1) ?: 0.0
|
||||||
|
this.formatter = formatter
|
||||||
|
this.save = save
|
||||||
|
buildView()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun buildView() {
|
||||||
|
val layout = view.findViewById<LinearLayout>(resLayoutId).also {
|
||||||
|
this.layout = it
|
||||||
|
it.removeAllViewsInLayout()
|
||||||
|
} ?: return
|
||||||
|
TextView(context).also {
|
||||||
|
this.textLabel = it
|
||||||
|
it.text = label
|
||||||
|
it.gravity = Gravity.CENTER
|
||||||
|
it.layoutParams = LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT).also { llp ->
|
||||||
|
llp.setMargins(0, 5, 0, 5)
|
||||||
|
}
|
||||||
|
TextViewCompat.setTextAppearance(it, android.R.style.TextAppearance_Medium)
|
||||||
|
layout.addView(it)
|
||||||
|
}
|
||||||
|
var i = 0
|
||||||
|
while (i < 24 && i < itemsCount()) {
|
||||||
|
inflateRow(i)
|
||||||
|
inflatedUntil = i
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
// last "plus" to append new interval
|
||||||
|
val factor = layout.context.resources.displayMetrics.density
|
||||||
|
ImageView(context).also {
|
||||||
|
this.finalAdd = it
|
||||||
|
it.setImageResource(info.nightscout.core.main.R.drawable.ic_add)
|
||||||
|
it.contentDescription = layout.context.resources.getString(info.nightscout.plugins.R.string.a11y_add_new_to_list)
|
||||||
|
layout.addView(it)
|
||||||
|
it.layoutParams = LinearLayout.LayoutParams((35.0 * factor).toInt(), (35 * factor).toInt()).also { llp ->
|
||||||
|
llp.setMargins(0, 25, 0, 25) // llp.setMargins(left, top, right, bottom);
|
||||||
|
llp.gravity = Gravity.CENTER
|
||||||
|
}
|
||||||
|
it.setOnClickListener {
|
||||||
|
addItem(itemsCount(), if (itemsCount() > 0) secondFromMidnight(itemsCount() - 1) + ONE_HOUR_IN_SECONDS else 0)
|
||||||
|
callSave()
|
||||||
|
log()
|
||||||
|
fillView()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fillView()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun inflateRow(position: Int) {
|
||||||
|
val resource =
|
||||||
|
if (data2 == null) info.nightscout.plugins.R.layout.timelistedit_element
|
||||||
|
else info.nightscout.plugins.R.layout.timelistedit_element_vertical
|
||||||
|
val childView = LayoutInflater.from(context).inflate(resource, layout, false).also {
|
||||||
|
intervals[position] = it
|
||||||
|
layout?.addView(it)
|
||||||
|
}
|
||||||
|
childView.findViewById<ImageView>(info.nightscout.plugins.R.id.timelistedit_add).also {
|
||||||
|
addButtons[position] = it
|
||||||
|
it.setOnClickListener {
|
||||||
|
val seconds = secondFromMidnight(position)
|
||||||
|
addItem(position, seconds)
|
||||||
|
// for here for the rest of values
|
||||||
|
for (i in position + 1 until itemsCount()) {
|
||||||
|
if (secondFromMidnight(i - 1) >= secondFromMidnight(i)) {
|
||||||
|
editItem(i, secondFromMidnight(i - 1) + ONE_HOUR_IN_SECONDS, value1(i), value2(i))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (itemsCount() > 24 || secondFromMidnight(itemsCount() - 1) > 23 * ONE_HOUR_IN_SECONDS) removeItem(itemsCount() - 1)
|
||||||
|
callSave()
|
||||||
|
log()
|
||||||
|
fillView()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
childView.findViewById<ImageView>(info.nightscout.plugins.R.id.timelistedit_remove).also {
|
||||||
|
removeButtons[position] = it
|
||||||
|
it.setOnClickListener {
|
||||||
|
removeItem(position)
|
||||||
|
callSave()
|
||||||
|
log()
|
||||||
|
fillView()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
SpinnerHelper(childView.findViewById(info.nightscout.plugins.R.id.timelistedit_time)).also {
|
||||||
|
spinners[position] = it
|
||||||
|
it.setOnItemSelectedListener(
|
||||||
|
object : AdapterView.OnItemSelectedListener {
|
||||||
|
override fun onItemSelected(parent: AdapterView<*>?, view: View, selected: Int, id: Long) {
|
||||||
|
val seconds = (it.adapter as SpinnerAdapter).valueForPosition(selected)
|
||||||
|
editItem(position, seconds, value1(position), value2(position))
|
||||||
|
log()
|
||||||
|
callSave()
|
||||||
|
fillView()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onNothingSelected(parent: AdapterView<*>?) {}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
childView.findViewById<NumberPicker>(info.nightscout.plugins.R.id.timelistedit_edit1).also {
|
||||||
|
numberPickers1[position] = it
|
||||||
|
it.setTextWatcher(object : TextWatcher {
|
||||||
|
override fun afterTextChanged(s: Editable) {
|
||||||
|
val value1 = stringToDouble(it.text, 0.0)
|
||||||
|
var value2 = value2(position)
|
||||||
|
if (data2 != null && value1 > value2) {
|
||||||
|
value2 = value1
|
||||||
|
numberPickers2[position]?.value = value2
|
||||||
|
}
|
||||||
|
editItem(position, secondFromMidnight(position), value1, value2)
|
||||||
|
callSave()
|
||||||
|
log()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {}
|
||||||
|
override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {}
|
||||||
|
})
|
||||||
|
it.tag = "$tagPrefix-1-$position"
|
||||||
|
}
|
||||||
|
childView.findViewById<NumberPicker>(info.nightscout.plugins.R.id.timelistedit_edit2).also {
|
||||||
|
numberPickers2[position] = it
|
||||||
|
it.setTextWatcher(object : TextWatcher {
|
||||||
|
override fun afterTextChanged(s: Editable) {
|
||||||
|
var value1 = value1(position)
|
||||||
|
val value2 = stringToDouble(it.text, 0.0)
|
||||||
|
if (data2 != null && value2 < value1) {
|
||||||
|
value1 = value2
|
||||||
|
numberPickers1[position]?.value = value1
|
||||||
|
}
|
||||||
|
editItem(position, secondFromMidnight(position), value1, value2)
|
||||||
|
callSave()
|
||||||
|
log()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) {}
|
||||||
|
override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) {}
|
||||||
|
})
|
||||||
|
it.tag = "$tagPrefix-2-$position"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun fillView() {
|
||||||
|
for (i in 0..23) {
|
||||||
|
if (i < itemsCount()) {
|
||||||
|
intervals[i]?.visibility = View.VISIBLE
|
||||||
|
buildInterval(i)
|
||||||
|
} else if (i <= inflatedUntil) {
|
||||||
|
intervals[i]?.visibility = View.GONE
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finalAdd?.visibility =
|
||||||
|
if (!(itemsCount() > 0 && secondFromMidnight(itemsCount() - 1) == 23 * ONE_HOUR_IN_SECONDS)) View.VISIBLE
|
||||||
|
else View.GONE
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun buildInterval(i: Int) {
|
||||||
|
val timeSpinner = spinners[i] ?: return
|
||||||
|
val editText1 = numberPickers1[i] ?: return
|
||||||
|
val editText2 = numberPickers2[i] ?: return
|
||||||
|
val previous = if (i == 0) -1 * ONE_HOUR_IN_SECONDS else secondFromMidnight(i - 1)
|
||||||
|
var next = if (i == itemsCount() - 1) 24 * ONE_HOUR_IN_SECONDS else secondFromMidnight(i + 1)
|
||||||
|
if (i == 0) next = ONE_HOUR_IN_SECONDS
|
||||||
|
fillSpinner(timeSpinner, secondFromMidnight(i), previous, next)
|
||||||
|
editText1.setParams(value1(i), min, max, step, formatter, false, null)
|
||||||
|
editText2.setParams(value2(i), min2, max2, step, formatter, false, null)
|
||||||
|
if (data2 == null) {
|
||||||
|
editText2.visibility = View.GONE
|
||||||
|
}
|
||||||
|
removeButtons[i]?.visibility =
|
||||||
|
if (itemsCount() == 1 || i == 0) View.INVISIBLE
|
||||||
|
else View.VISIBLE
|
||||||
|
addButtons[i]?.visibility =
|
||||||
|
if (itemsCount() >= 24 || secondFromMidnight(i) >= 82800) View.INVISIBLE
|
||||||
|
else View.VISIBLE
|
||||||
|
}
|
||||||
|
|
||||||
|
internal class SpinnerAdapter(context: Context, resource: Int, objects: List<CharSequence>, var values: List<Int>) : ArrayAdapter<CharSequence?>(context, resource, objects) {
|
||||||
|
|
||||||
|
fun valueForPosition(position: Int): Int = values[position]
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun fillSpinner(spinner: SpinnerHelper, secondsFromMidnight: Int, previous: Int, next: Int) {
|
||||||
|
var posInList = 0
|
||||||
|
val timeList = ArrayList<CharSequence>()
|
||||||
|
val timeListValues = ArrayList<Int>()
|
||||||
|
var pos = 0
|
||||||
|
var t = previous + ONE_HOUR_IN_SECONDS
|
||||||
|
while (t < next) {
|
||||||
|
timeList.add(dateUtil.timeStringFromSeconds(t))
|
||||||
|
timeListValues.add(t)
|
||||||
|
if (secondsFromMidnight == t) posInList = pos
|
||||||
|
pos++
|
||||||
|
t += ONE_HOUR_IN_SECONDS
|
||||||
|
}
|
||||||
|
val adapter = SpinnerAdapter(
|
||||||
|
context,
|
||||||
|
info.nightscout.core.ui.R.layout.spinner_centered, timeList, timeListValues
|
||||||
|
)
|
||||||
|
spinner.adapter = adapter
|
||||||
|
spinner.setSelection(posInList, false)
|
||||||
|
adapter.notifyDataSetChanged()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun itemsCount(): Int {
|
||||||
|
return data1.length()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun secondFromMidnight(index: Int): Int {
|
||||||
|
try {
|
||||||
|
val item = data1[index] as JSONObject
|
||||||
|
if (item.has("timeAsSeconds")) {
|
||||||
|
var time = item.getInt("timeAsSeconds")
|
||||||
|
if (index == 0 && time != 0) {
|
||||||
|
// fix the bug, every array must start with 0
|
||||||
|
item.put("timeAsSeconds", 0)
|
||||||
|
time = 0
|
||||||
|
}
|
||||||
|
return time
|
||||||
|
}
|
||||||
|
} catch (e: JSONException) {
|
||||||
|
aapsLogger.error("Unhandled exception", e)
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun value1(index: Int): Double {
|
||||||
|
try {
|
||||||
|
val item = data1[index] as JSONObject
|
||||||
|
if (item.has("value")) {
|
||||||
|
return item.getDouble("value")
|
||||||
|
}
|
||||||
|
} catch (e: JSONException) {
|
||||||
|
aapsLogger.error("Unhandled exception", e)
|
||||||
|
}
|
||||||
|
return 0.0
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun value2(index: Int): Double {
|
||||||
|
if (data2 != null) {
|
||||||
|
try {
|
||||||
|
val item = data2[index] as JSONObject
|
||||||
|
if (item.has("value")) {
|
||||||
|
return item.getDouble("value")
|
||||||
|
}
|
||||||
|
} catch (e: JSONException) {
|
||||||
|
aapsLogger.error("Unhandled exception", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0.0
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun editItem(index: Int, timeAsSeconds: Int, value1: Double, value2: Double) {
|
||||||
|
try {
|
||||||
|
val time: String
|
||||||
|
val hour = timeAsSeconds / 60 / 60
|
||||||
|
val df = DecimalFormat("00")
|
||||||
|
time = df.format(hour.toLong()) + ":00"
|
||||||
|
val newObject1 = JSONObject()
|
||||||
|
newObject1.put("time", time)
|
||||||
|
newObject1.put("timeAsSeconds", timeAsSeconds)
|
||||||
|
newObject1.put("value", value1)
|
||||||
|
data1.put(index, newObject1)
|
||||||
|
if (data2 != null) {
|
||||||
|
val newObject2 = JSONObject()
|
||||||
|
newObject2.put("time", time)
|
||||||
|
newObject2.put("timeAsSeconds", timeAsSeconds)
|
||||||
|
newObject2.put("value", value2)
|
||||||
|
data2.put(index, newObject2)
|
||||||
|
}
|
||||||
|
} catch (e: JSONException) {
|
||||||
|
aapsLogger.error("Unhandled exception", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun addItem(index: Int, timeAsSeconds: Int) {
|
||||||
|
if (itemsCount() >= 24) return
|
||||||
|
if (itemsCount() > inflatedUntil) {
|
||||||
|
layout?.removeView(finalAdd)
|
||||||
|
inflateRow(++inflatedUntil)
|
||||||
|
layout?.addView(finalAdd)
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// shift data
|
||||||
|
for (i in data1.length() downTo index + 1) {
|
||||||
|
data1.put(i, data1[i - 1])
|
||||||
|
data2?.put(i, data2[i - 1])
|
||||||
|
}
|
||||||
|
// add new object
|
||||||
|
editItem(index, timeAsSeconds, 0.0, 0.0)
|
||||||
|
} catch (e: JSONException) {
|
||||||
|
aapsLogger.error("Unhandled exception", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun removeItem(index: Int) {
|
||||||
|
data1.remove(index)
|
||||||
|
data2?.remove(index)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun log() {
|
||||||
|
for (i in 0 until data1.length()) {
|
||||||
|
aapsLogger.debug(i.toString() + ": @" + dateUtil.timeStringFromSeconds(secondFromMidnight(i)) + " " + value1(i) + if (data2 != null) " " + value2(i) else "")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun callSave() {
|
||||||
|
save?.run()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun updateLabel(txt: String) {
|
||||||
|
label = txt
|
||||||
|
textLabel?.text = txt
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
|
||||||
|
private const val ONE_HOUR_IN_SECONDS = 60 * 60
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,437 +0,0 @@
|
||||||
package info.nightscout.plugins.ui;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.text.Editable;
|
|
||||||
import android.text.TextWatcher;
|
|
||||||
import android.view.Gravity;
|
|
||||||
import android.view.LayoutInflater;
|
|
||||||
import android.view.View;
|
|
||||||
import android.widget.AdapterView;
|
|
||||||
import android.widget.ArrayAdapter;
|
|
||||||
import android.widget.ImageView;
|
|
||||||
import android.widget.LinearLayout;
|
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
|
||||||
import androidx.core.widget.TextViewCompat;
|
|
||||||
|
|
||||||
import org.json.JSONArray;
|
|
||||||
import org.json.JSONException;
|
|
||||||
import org.json.JSONObject;
|
|
||||||
|
|
||||||
import java.text.DecimalFormat;
|
|
||||||
import java.text.NumberFormat;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import info.nightscout.core.ui.elements.NumberPicker;
|
|
||||||
import info.nightscout.core.ui.elements.SpinnerHelper;
|
|
||||||
import info.nightscout.plugins.R;
|
|
||||||
import info.nightscout.rx.logging.AAPSLogger;
|
|
||||||
import info.nightscout.shared.SafeParse;
|
|
||||||
import info.nightscout.shared.utils.DateUtil;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Created by mike on 29.12.2016.
|
|
||||||
*/
|
|
||||||
|
|
||||||
public class TimeListEdit {
|
|
||||||
private final AAPSLogger aapsLogger;
|
|
||||||
private final DateUtil dateUtil;
|
|
||||||
|
|
||||||
private final int ONEHOURINSECONDS = 60 * 60;
|
|
||||||
|
|
||||||
private final View[] intervals = new View[24];
|
|
||||||
private final SpinnerHelper[] spinners = new SpinnerHelper[24];
|
|
||||||
private final NumberPicker[] numberPickers1 = new NumberPicker[24];
|
|
||||||
private final NumberPicker[] numberPickers2 = new NumberPicker[24];
|
|
||||||
private final ImageView[] addButtons = new ImageView[24];
|
|
||||||
private final ImageView[] removeButtons = new ImageView[24];
|
|
||||||
private ImageView finalAdd;
|
|
||||||
|
|
||||||
private final Context context;
|
|
||||||
private final View view;
|
|
||||||
private final int resLayoutId;
|
|
||||||
private final String tagPrefix;
|
|
||||||
private String label;
|
|
||||||
private final JSONArray data1;
|
|
||||||
private final JSONArray data2;
|
|
||||||
private final double step;
|
|
||||||
private final double min;
|
|
||||||
private final double max;
|
|
||||||
private final double min2;
|
|
||||||
private final double max2;
|
|
||||||
private final NumberFormat formatter;
|
|
||||||
private final Runnable save;
|
|
||||||
private LinearLayout layout;
|
|
||||||
private TextView textlabel;
|
|
||||||
private int inflatedUntil = -1;
|
|
||||||
|
|
||||||
|
|
||||||
public TimeListEdit(
|
|
||||||
Context context,
|
|
||||||
AAPSLogger aapsLogger,
|
|
||||||
DateUtil dateUtil,
|
|
||||||
View view, int resLayoutId, String tagPrefix, String label, JSONArray data1, JSONArray data2, double[] range1, double[] range2, double step, NumberFormat formatter, Runnable save) {
|
|
||||||
this.context = context;
|
|
||||||
this.aapsLogger = aapsLogger;
|
|
||||||
this.dateUtil = dateUtil;
|
|
||||||
this.view = view;
|
|
||||||
this.resLayoutId = resLayoutId;
|
|
||||||
this.tagPrefix = tagPrefix;
|
|
||||||
this.label = label;
|
|
||||||
this.data1 = data1;
|
|
||||||
this.data2 = data2;
|
|
||||||
this.step = step;
|
|
||||||
this.min = range1[0];
|
|
||||||
this.max = range1[1];
|
|
||||||
this.min2 = range2 != null ? range2[0] : 0;
|
|
||||||
this.max2 = range2 != null ? range2[1] : 0;
|
|
||||||
this.formatter = formatter;
|
|
||||||
this.save = save;
|
|
||||||
buildView();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void buildView() {
|
|
||||||
layout = view.findViewById(resLayoutId);
|
|
||||||
layout.removeAllViewsInLayout();
|
|
||||||
|
|
||||||
textlabel = new TextView(context);
|
|
||||||
textlabel.setText(label);
|
|
||||||
textlabel.setGravity(Gravity.CENTER);
|
|
||||||
LinearLayout.LayoutParams llp = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT);
|
|
||||||
llp.setMargins(0, 5, 0, 5);
|
|
||||||
textlabel.setLayoutParams(llp);
|
|
||||||
TextViewCompat.setTextAppearance(textlabel, android.R.style.TextAppearance_Medium);
|
|
||||||
layout.addView(textlabel);
|
|
||||||
|
|
||||||
for (int i = 0; i < 24 && i < itemsCount(); i++) {
|
|
||||||
inflateRow(i);
|
|
||||||
inflatedUntil = i;
|
|
||||||
}
|
|
||||||
|
|
||||||
// last "plus" to append new interval
|
|
||||||
float factor = layout.getContext().getResources().getDisplayMetrics().density;
|
|
||||||
finalAdd = new ImageView(context);
|
|
||||||
finalAdd.setImageResource(info.nightscout.core.main.R.drawable.ic_add);
|
|
||||||
finalAdd.setContentDescription(layout.getContext().getResources().getString(R.string.a11y_add_new_to_list));
|
|
||||||
LinearLayout.LayoutParams illp = new LinearLayout.LayoutParams((int) (35d * factor), (int) (35 * factor));
|
|
||||||
illp.setMargins(0, 25, 0, 25); // llp.setMargins(left, top, right, bottom);
|
|
||||||
illp.gravity = Gravity.CENTER;
|
|
||||||
layout.addView(finalAdd);
|
|
||||||
finalAdd.setLayoutParams(illp);
|
|
||||||
finalAdd.setOnClickListener(view -> {
|
|
||||||
addItem(itemsCount(), itemsCount() > 0 ? secondFromMidnight(itemsCount() - 1) + ONEHOURINSECONDS : 0, 0, 0);
|
|
||||||
callSave();
|
|
||||||
log();
|
|
||||||
fillView();
|
|
||||||
});
|
|
||||||
|
|
||||||
fillView();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void inflateRow(final int position) {
|
|
||||||
|
|
||||||
LayoutInflater inflater = LayoutInflater.from(context);
|
|
||||||
int resource = data2 == null ? R.layout.timelistedit_element : R.layout.timelistedit_element_vertical;
|
|
||||||
View childView = intervals[position] = inflater.inflate(resource, layout, false);
|
|
||||||
spinners[position] = new SpinnerHelper(childView.findViewById(R.id.timelistedit_time));
|
|
||||||
numberPickers1[position] = childView.findViewById(R.id.timelistedit_edit1);
|
|
||||||
numberPickers2[position] = childView.findViewById(R.id.timelistedit_edit2);
|
|
||||||
addButtons[position] = childView.findViewById(R.id.timelistedit_add);
|
|
||||||
removeButtons[position] = childView.findViewById(R.id.timelistedit_remove);
|
|
||||||
|
|
||||||
addButtons[position].setOnClickListener(view -> {
|
|
||||||
int seconds = secondFromMidnight(position);
|
|
||||||
addItem(position, seconds, 0, 0);
|
|
||||||
// for here for the rest of values
|
|
||||||
for (int i = position + 1; i < itemsCount(); i++) {
|
|
||||||
if (secondFromMidnight(i - 1) >= secondFromMidnight(i)) {
|
|
||||||
editItem(i, secondFromMidnight(i - 1) + ONEHOURINSECONDS, value1(i), value2(i));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while (itemsCount() > 24 || secondFromMidnight(itemsCount() - 1) > 23 * ONEHOURINSECONDS)
|
|
||||||
removeItem(itemsCount() - 1);
|
|
||||||
callSave();
|
|
||||||
log();
|
|
||||||
fillView();
|
|
||||||
});
|
|
||||||
|
|
||||||
removeButtons[position].setOnClickListener(view -> {
|
|
||||||
removeItem(position);
|
|
||||||
callSave();
|
|
||||||
log();
|
|
||||||
fillView();
|
|
||||||
});
|
|
||||||
|
|
||||||
spinners[position].setOnItemSelectedListener(
|
|
||||||
new AdapterView.OnItemSelectedListener() {
|
|
||||||
@Override
|
|
||||||
public void onItemSelected(AdapterView<?> parent, View view, int selected, long id) {
|
|
||||||
int seconds = ((SpinnerAdapter) spinners[position].getAdapter()).valueForPosition(selected);
|
|
||||||
editItem(position, seconds, value1(position), value2(position));
|
|
||||||
log();
|
|
||||||
callSave();
|
|
||||||
fillView();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onNothingSelected(AdapterView<?> parent) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
numberPickers1[position].setTextWatcher(new TextWatcher() {
|
|
||||||
@Override
|
|
||||||
public void afterTextChanged(Editable s) {
|
|
||||||
Double value1 = SafeParse.INSTANCE.stringToDouble(numberPickers1[position].getText(), 0.0);
|
|
||||||
Double value2 = value2(position);
|
|
||||||
if (data2 != null && value1 > value2) {
|
|
||||||
value2 = value1;
|
|
||||||
numberPickers2[position].setValue(value2);
|
|
||||||
}
|
|
||||||
editItem(position, secondFromMidnight(position), value1, value2);
|
|
||||||
callSave();
|
|
||||||
log();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void beforeTextChanged(CharSequence s, int start,
|
|
||||||
int count, int after) {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onTextChanged(CharSequence s, int start,
|
|
||||||
int before, int count) {
|
|
||||||
}
|
|
||||||
});
|
|
||||||
numberPickers1[position].setTag(tagPrefix + "-1-" + position);
|
|
||||||
|
|
||||||
numberPickers2[position].setTextWatcher(new TextWatcher() {
|
|
||||||
@Override
|
|
||||||
public void afterTextChanged(Editable s) {
|
|
||||||
Double value1 = value1(position);
|
|
||||||
Double value2 = SafeParse.INSTANCE.stringToDouble(numberPickers2[position].getText(), 0.0);
|
|
||||||
if (data2 != null && value2 < value1) {
|
|
||||||
value1 = value2;
|
|
||||||
numberPickers1[position].setValue(value1);
|
|
||||||
}
|
|
||||||
editItem(position, secondFromMidnight(position), value1, value2);
|
|
||||||
callSave();
|
|
||||||
log();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void beforeTextChanged(CharSequence s, int start,
|
|
||||||
int count, int after) {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onTextChanged(CharSequence s, int start,
|
|
||||||
int before, int count) {
|
|
||||||
}
|
|
||||||
});
|
|
||||||
numberPickers2[position].setTag(tagPrefix + "-2-" + position);
|
|
||||||
|
|
||||||
layout.addView(childView);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void fillView() {
|
|
||||||
for (int i = 0; i < 24; i++) {
|
|
||||||
if (i < itemsCount()) {
|
|
||||||
intervals[i].setVisibility(View.VISIBLE);
|
|
||||||
buildInterval(i);
|
|
||||||
} else if (i <= inflatedUntil) {
|
|
||||||
intervals[i].setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(itemsCount() > 0 && secondFromMidnight(itemsCount() - 1) == 23 * ONEHOURINSECONDS)) {
|
|
||||||
finalAdd.setVisibility(View.VISIBLE);
|
|
||||||
} else {
|
|
||||||
finalAdd.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void buildInterval(int i) {
|
|
||||||
SpinnerHelper timeSpinner = spinners[i];
|
|
||||||
final NumberPicker editText1 = numberPickers1[i];
|
|
||||||
final NumberPicker editText2 = numberPickers2[i];
|
|
||||||
|
|
||||||
|
|
||||||
int previous = i == 0 ? -1 * ONEHOURINSECONDS : secondFromMidnight(i - 1);
|
|
||||||
int next = i == itemsCount() - 1 ? 24 * ONEHOURINSECONDS : secondFromMidnight(i + 1);
|
|
||||||
if (i == 0) next = ONEHOURINSECONDS;
|
|
||||||
fillSpinner(timeSpinner, secondFromMidnight(i), previous, next);
|
|
||||||
|
|
||||||
editText1.setParams(value1(i), min, max, step, formatter, false, null);
|
|
||||||
editText2.setParams(value2(i), min2, max2, step, formatter, false, null);
|
|
||||||
|
|
||||||
if (data2 == null) {
|
|
||||||
editText2.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (itemsCount() == 1 || i == 0) {
|
|
||||||
removeButtons[i].setVisibility(View.INVISIBLE);
|
|
||||||
} else
|
|
||||||
removeButtons[i].setVisibility(View.VISIBLE);
|
|
||||||
|
|
||||||
if (itemsCount() >= 24 || secondFromMidnight(i) >= 82800) {
|
|
||||||
addButtons[i].setVisibility(View.INVISIBLE);
|
|
||||||
} else {
|
|
||||||
addButtons[i].setVisibility(View.VISIBLE);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
static class SpinnerAdapter extends ArrayAdapter<CharSequence> {
|
|
||||||
List<Integer> values;
|
|
||||||
|
|
||||||
SpinnerAdapter(@NonNull Context context, int resource, final @NonNull List<CharSequence> objects, final @NonNull List<Integer> values) {
|
|
||||||
super(context, resource, objects);
|
|
||||||
this.values = values;
|
|
||||||
}
|
|
||||||
|
|
||||||
int valueForPosition(int position) {
|
|
||||||
return values.get(position);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void fillSpinner(final SpinnerHelper spinner, int secondsFromMidnight, int previous, int next) {
|
|
||||||
int posInList = 0;
|
|
||||||
ArrayList<CharSequence> timeList = new ArrayList<>();
|
|
||||||
ArrayList<Integer> timeListValues = new ArrayList<>();
|
|
||||||
int pos = 0;
|
|
||||||
for (int t = previous + ONEHOURINSECONDS; t < next; t += ONEHOURINSECONDS) {
|
|
||||||
timeList.add(dateUtil.timeStringFromSeconds(t));
|
|
||||||
timeListValues.add(t);
|
|
||||||
if (secondsFromMidnight == t) posInList = pos;
|
|
||||||
pos++;
|
|
||||||
}
|
|
||||||
|
|
||||||
final SpinnerAdapter adapter = new SpinnerAdapter(context,
|
|
||||||
info.nightscout.core.ui.R.layout.spinner_centered, timeList, timeListValues);
|
|
||||||
spinner.setAdapter(adapter);
|
|
||||||
spinner.setSelection(posInList, false);
|
|
||||||
adapter.notifyDataSetChanged();
|
|
||||||
}
|
|
||||||
|
|
||||||
private int itemsCount() {
|
|
||||||
return data1.length();
|
|
||||||
}
|
|
||||||
|
|
||||||
private int secondFromMidnight(int index) {
|
|
||||||
try {
|
|
||||||
JSONObject item = (JSONObject) data1.get(index);
|
|
||||||
if (item.has("timeAsSeconds")) {
|
|
||||||
int time = item.getInt("timeAsSeconds");
|
|
||||||
if (index == 0 && time != 0) {
|
|
||||||
// fix the bug, every array must start with 0
|
|
||||||
item.put("timeAsSeconds", 0);
|
|
||||||
time = 0;
|
|
||||||
}
|
|
||||||
return time;
|
|
||||||
}
|
|
||||||
} catch (JSONException e) {
|
|
||||||
aapsLogger.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
private double value1(int index) {
|
|
||||||
try {
|
|
||||||
JSONObject item = (JSONObject) data1.get(index);
|
|
||||||
if (item.has("value")) {
|
|
||||||
return item.getDouble("value");
|
|
||||||
}
|
|
||||||
} catch (JSONException e) {
|
|
||||||
aapsLogger.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
return 0d;
|
|
||||||
}
|
|
||||||
|
|
||||||
private double value2(int index) {
|
|
||||||
if (data2 != null) {
|
|
||||||
try {
|
|
||||||
JSONObject item = (JSONObject) data2.get(index);
|
|
||||||
if (item.has("value")) {
|
|
||||||
return item.getDouble("value");
|
|
||||||
}
|
|
||||||
} catch (JSONException e) {
|
|
||||||
aapsLogger.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0d;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void editItem(int index, int timeAsSeconds, double value1, double value2) {
|
|
||||||
try {
|
|
||||||
String time;
|
|
||||||
int hour = timeAsSeconds / 60 / 60;
|
|
||||||
DecimalFormat df = new DecimalFormat("00");
|
|
||||||
time = df.format(hour) + ":00";
|
|
||||||
|
|
||||||
JSONObject newObject1 = new JSONObject();
|
|
||||||
newObject1.put("time", time);
|
|
||||||
newObject1.put("timeAsSeconds", timeAsSeconds);
|
|
||||||
newObject1.put("value", value1);
|
|
||||||
data1.put(index, newObject1);
|
|
||||||
if (data2 != null) {
|
|
||||||
JSONObject newObject2 = new JSONObject();
|
|
||||||
newObject2.put("time", time);
|
|
||||||
newObject2.put("timeAsSeconds", timeAsSeconds);
|
|
||||||
newObject2.put("value", value2);
|
|
||||||
data2.put(index, newObject2);
|
|
||||||
}
|
|
||||||
} catch (JSONException e) {
|
|
||||||
aapsLogger.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@SuppressWarnings("SameParameterValue")
|
|
||||||
private void addItem(int index, int timeAsSeconds, double value1, double value2) {
|
|
||||||
if (itemsCount() >= 24) return;
|
|
||||||
if (itemsCount() > inflatedUntil) {
|
|
||||||
layout.removeView(finalAdd);
|
|
||||||
inflateRow(++inflatedUntil);
|
|
||||||
layout.addView(finalAdd);
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
// shift data
|
|
||||||
for (int i = data1.length(); i > index; i--) {
|
|
||||||
data1.put(i, data1.get(i - 1));
|
|
||||||
if (data2 != null)
|
|
||||||
data2.put(i, data2.get(i - 1));
|
|
||||||
}
|
|
||||||
// add new object
|
|
||||||
editItem(index, timeAsSeconds, value1, value2);
|
|
||||||
} catch (JSONException e) {
|
|
||||||
aapsLogger.error("Unhandled exception", e);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private void removeItem(int index) {
|
|
||||||
data1.remove(index);
|
|
||||||
if (data2 != null)
|
|
||||||
data2.remove(index);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void log() {
|
|
||||||
for (int i = 0; i < data1.length(); i++) {
|
|
||||||
aapsLogger.debug(i + ": @" + dateUtil.timeStringFromSeconds(secondFromMidnight(i)) + " " + value1(i) + (data2 != null ? " " + value2(i) : ""));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void callSave() {
|
|
||||||
if (save != null) save.run();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void updateLabel(String txt) {
|
|
||||||
this.label = txt;
|
|
||||||
if (textlabel != null)
|
|
||||||
textlabel.setText(txt);
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in a new issue