mirror of
https://github.com/meshtastic/Meshtastic-Android.git
synced 2026-04-20 22:23:37 +00:00
197 lines
8.6 KiB
Kotlin
197 lines
8.6 KiB
Kotlin
/*
|
|
* Copyright (c) 2025 Meshtastic LLC
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
package com.geeksville.mesh.model
|
|
|
|
import androidx.compose.ui.graphics.Color
|
|
import com.geeksville.mesh.TelemetryProtos
|
|
import org.meshtastic.core.model.util.UnitConversions
|
|
import org.meshtastic.core.ui.theme.GraphColors.Green
|
|
import org.meshtastic.core.ui.theme.GraphColors.InfantryBlue
|
|
import org.meshtastic.core.ui.theme.GraphColors.LightGreen
|
|
import org.meshtastic.core.ui.theme.GraphColors.Magenta
|
|
import org.meshtastic.core.ui.theme.GraphColors.Orange
|
|
import org.meshtastic.core.ui.theme.GraphColors.Pink
|
|
import org.meshtastic.core.ui.theme.GraphColors.Purple
|
|
import org.meshtastic.core.ui.theme.GraphColors.Red
|
|
import org.meshtastic.core.ui.theme.GraphColors.Yellow
|
|
|
|
@Suppress("MagicNumber")
|
|
enum class Environment(val color: Color) {
|
|
TEMPERATURE(Red) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.temperature
|
|
},
|
|
HUMIDITY(InfantryBlue) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.relativeHumidity
|
|
},
|
|
SOIL_TEMPERATURE(Pink) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.soilTemperature
|
|
},
|
|
SOIL_MOISTURE(Purple) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) =
|
|
telemetry.environmentMetrics.soilMoisture?.toFloat()
|
|
},
|
|
BAROMETRIC_PRESSURE(Green) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.barometricPressure
|
|
},
|
|
GAS_RESISTANCE(Yellow) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.gasResistance
|
|
},
|
|
IAQ(Magenta) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.iaq?.toFloat()
|
|
},
|
|
LUX(LightGreen) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.lux
|
|
},
|
|
UV_LUX(Orange) {
|
|
override fun getValue(telemetry: TelemetryProtos.Telemetry) = telemetry.environmentMetrics.uvLux
|
|
}, ;
|
|
|
|
abstract fun getValue(telemetry: TelemetryProtos.Telemetry): Float?
|
|
}
|
|
|
|
/**
|
|
* @param metrics the filtered [List]
|
|
* @param shouldPlot a [List] the size of [Environment] used to determine if a metric should be plotted
|
|
* @param leftMinMax [Pair] with the min and max of the barometric pressure
|
|
* @param rightMinMax [Pair] with the combined min and max of: the temperature, humidity, and IAQ
|
|
* @param times [Pair] with the oldest and newest times in that order
|
|
*/
|
|
data class EnvironmentGraphingData(
|
|
val metrics: List<TelemetryProtos.Telemetry>,
|
|
val shouldPlot: List<Boolean>,
|
|
val leftMinMax: Pair<Float, Float> = Pair(0f, 0f),
|
|
val rightMinMax: Pair<Float, Float> = Pair(0f, 0f),
|
|
val times: Pair<Int, Int> = Pair(0, 0),
|
|
)
|
|
|
|
data class EnvironmentMetricsState(val environmentMetrics: List<TelemetryProtos.Telemetry> = emptyList()) {
|
|
fun hasEnvironmentMetrics() = environmentMetrics.isNotEmpty()
|
|
|
|
/**
|
|
* Filters [environmentMetrics] based on a [TimeFrame].
|
|
*
|
|
* @param timeFrame used to filter
|
|
* @return [EnvironmentGraphingData]
|
|
*/
|
|
@Suppress("LongMethod", "CyclomaticComplexMethod", "MagicNumber")
|
|
fun environmentMetricsFiltered(timeFrame: TimeFrame, useFahrenheit: Boolean = false): EnvironmentGraphingData {
|
|
val oldestTime = timeFrame.calculateOldestTime()
|
|
val telemetries = environmentMetrics.filter { it.time >= oldestTime }
|
|
val shouldPlot = BooleanArray(Environment.entries.size) { false }
|
|
if (telemetries.isEmpty()) {
|
|
return EnvironmentGraphingData(metrics = telemetries, shouldPlot = shouldPlot.toList())
|
|
}
|
|
|
|
/* Grab the combined min and max for temp, humidity, soil_Temperature, soilMoisture and iaq. */
|
|
val minValues = mutableListOf<Float>()
|
|
val maxValues = mutableListOf<Float>()
|
|
|
|
// Temperature
|
|
val temperatures = telemetries.mapNotNull { it.environmentMetrics.temperature?.takeIf { !it.isNaN() } }
|
|
if (temperatures.isNotEmpty()) {
|
|
var minTempValue = temperatures.minOf { it }
|
|
var maxTempValue = temperatures.maxOf { it }
|
|
if (useFahrenheit) {
|
|
minTempValue = UnitConversions.celsiusToFahrenheit(minTempValue)
|
|
maxTempValue = UnitConversions.celsiusToFahrenheit(maxTempValue)
|
|
}
|
|
minValues.add(minTempValue)
|
|
maxValues.add(maxTempValue)
|
|
shouldPlot[Environment.TEMPERATURE.ordinal] = true
|
|
}
|
|
|
|
// Relative Humidity
|
|
val humidities =
|
|
telemetries.mapNotNull { it.environmentMetrics.relativeHumidity?.takeIf { !it.isNaN() && it != 0.0f } }
|
|
if (humidities.isNotEmpty()) {
|
|
minValues.add(humidities.minOf { it })
|
|
maxValues.add(humidities.maxOf { it })
|
|
shouldPlot[Environment.HUMIDITY.ordinal] = true
|
|
}
|
|
|
|
// Soil Temperature
|
|
val soilTemperatures = telemetries.mapNotNull { it.environmentMetrics.soilTemperature?.takeIf { !it.isNaN() } }
|
|
if (soilTemperatures.isNotEmpty()) {
|
|
var minSoilTemperatureValue = soilTemperatures.minOf { it }
|
|
var maxSoilTemperatureValue = soilTemperatures.maxOf { it }
|
|
if (useFahrenheit) {
|
|
minSoilTemperatureValue = UnitConversions.celsiusToFahrenheit(minSoilTemperatureValue)
|
|
maxSoilTemperatureValue = UnitConversions.celsiusToFahrenheit(maxSoilTemperatureValue)
|
|
}
|
|
minValues.add(minSoilTemperatureValue)
|
|
maxValues.add(maxSoilTemperatureValue)
|
|
shouldPlot[Environment.SOIL_TEMPERATURE.ordinal] = true
|
|
}
|
|
|
|
// Soil Moisture
|
|
val soilMoistures =
|
|
telemetries.mapNotNull { it.environmentMetrics.soilMoisture?.takeIf { it != Int.MIN_VALUE } }
|
|
if (soilMoistures.isNotEmpty()) {
|
|
minValues.add(soilMoistures.minOf { it.toFloat() })
|
|
maxValues.add(soilMoistures.maxOf { it.toFloat() })
|
|
shouldPlot[Environment.SOIL_MOISTURE.ordinal] = true
|
|
}
|
|
|
|
// IAQ
|
|
val iaqs = telemetries.mapNotNull { it.environmentMetrics.iaq?.takeIf { it != Int.MIN_VALUE } }
|
|
if (iaqs.isNotEmpty()) {
|
|
minValues.add(iaqs.minOf { it.toFloat() })
|
|
maxValues.add(iaqs.maxOf { it.toFloat() })
|
|
shouldPlot[Environment.IAQ.ordinal] = true
|
|
}
|
|
|
|
// Barometric Pressure
|
|
val pressures = telemetries.mapNotNull { it.environmentMetrics.barometricPressure?.takeIf { !it.isNaN() } }
|
|
var minPressureValue = 0f
|
|
var maxPressureValue = 0f
|
|
if (pressures.isNotEmpty()) {
|
|
minPressureValue = pressures.minOf { it }
|
|
maxPressureValue = pressures.maxOf { it }
|
|
shouldPlot[Environment.BAROMETRIC_PRESSURE.ordinal] = true
|
|
}
|
|
|
|
// Lux
|
|
val luxValues = telemetries.mapNotNull { it.environmentMetrics.lux?.takeIf { !it.isNaN() } }
|
|
if (luxValues.isNotEmpty()) {
|
|
minValues.add(luxValues.minOf { it })
|
|
maxValues.add(luxValues.maxOf { it })
|
|
shouldPlot[Environment.LUX.ordinal] = true
|
|
}
|
|
|
|
// UVLux
|
|
val uvLuxValues = telemetries.mapNotNull { it.environmentMetrics.uvLux?.takeIf { !it.isNaN() } }
|
|
if (uvLuxValues.isNotEmpty()) {
|
|
minValues.add(uvLuxValues.minOf { it })
|
|
maxValues.add(uvLuxValues.maxOf { it })
|
|
shouldPlot[Environment.UV_LUX.ordinal] = true
|
|
}
|
|
|
|
val min = if (minValues.isEmpty()) 0f else minValues.minOf { it }
|
|
val max = if (maxValues.isEmpty()) 1f else maxValues.maxOf { it }
|
|
|
|
val (oldest, newest) = Pair(telemetries.minBy { it.time }, telemetries.maxBy { it.time })
|
|
|
|
return EnvironmentGraphingData(
|
|
metrics = telemetries,
|
|
shouldPlot = shouldPlot.toList(),
|
|
leftMinMax = Pair(minPressureValue, maxPressureValue),
|
|
rightMinMax = Pair(min, max),
|
|
times = Pair(oldest.time, newest.time),
|
|
)
|
|
}
|
|
}
|