refactor: Node detail work (#1836)

This commit is contained in:
Robert-0410 2025-05-10 06:03:14 -07:00 committed by GitHub
parent 6bb282f901
commit cd9167f19f
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 96 additions and 70 deletions

View file

@ -15,15 +15,12 @@
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
@file:Suppress("TooManyFunctions")
package com.geeksville.mesh.ui
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.ExperimentalLayoutApi
import androidx.compose.foundation.layout.FlowRow
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Row
@ -93,7 +90,6 @@ import androidx.compose.ui.unit.dp
import androidx.hilt.navigation.compose.hiltViewModel
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import coil3.compose.AsyncImage
import com.geeksville.mesh.ConfigProtos.Config.DisplayConfig.DisplayUnits
import com.geeksville.mesh.R
import com.geeksville.mesh.model.DeviceHardware
import com.geeksville.mesh.model.MetricsState
@ -104,11 +100,12 @@ import com.geeksville.mesh.ui.components.PreferenceCategory
import com.geeksville.mesh.ui.preview.NodePreviewParameterProvider
import com.geeksville.mesh.ui.radioconfig.NavCard
import com.geeksville.mesh.ui.theme.AppTheme
import com.geeksville.mesh.util.DistanceUnit
import com.geeksville.mesh.util.UnitConversions.calculateDewPoint
import com.geeksville.mesh.util.UnitConversions.toTempString
import com.geeksville.mesh.util.formatAgo
import com.geeksville.mesh.util.formatUptime
import com.geeksville.mesh.util.thenIf
import kotlin.math.ln
import com.geeksville.mesh.util.toSpeedString
private enum class LogsType(
val titleRes: Int,
@ -286,13 +283,13 @@ private fun DeviceDetailsContent(
)
}
NodeDetailRow(
label = "Hardware",
label = stringResource(R.string.hardware),
icon = Icons.Default.Router,
value = hwModelName
)
if (isSupported) {
NodeDetailRow(
label = "Supported",
label = stringResource(R.string.supported),
icon = Icons.Default.Verified,
value = "",
iconTint = Color.Green
@ -349,36 +346,36 @@ private fun NodeDetailsContent(
Spacer(Modifier.height(16.dp))
}
NodeDetailRow(
label = "Node Number",
label = stringResource(R.string.node_number),
icon = Icons.Default.Numbers,
value = node.num.toUInt().toString()
)
NodeDetailRow(
label = "User Id",
label = stringResource(R.string.user_id),
icon = Icons.Default.Person,
value = node.user.id
)
NodeDetailRow(
label = "Role",
label = stringResource(R.string.role),
icon = Icons.Default.Work,
value = node.user.role.name
)
if (node.deviceMetrics.uptimeSeconds > 0) {
NodeDetailRow(
label = "Uptime",
label = stringResource(R.string.uptime),
icon = Icons.Default.CheckCircle,
value = formatUptime(node.deviceMetrics.uptimeSeconds)
)
}
if (node.metadata != null) {
NodeDetailRow(
label = "Firmware version",
label = stringResource(R.string.firmware_version),
icon = Icons.Default.Memory,
value = node.metadata.firmwareVersion.substringBeforeLast(".")
)
}
NodeDetailRow(
label = "Last heard",
label = stringResource(R.string.node_sort_last_heard),
icon = Icons.Default.History,
value = formatAgo(node.lastHeard)
)
@ -431,7 +428,6 @@ private fun InfoCard(
}
}
@OptIn(ExperimentalLayoutApi::class)
@Suppress("LongMethod", "CyclomaticComplexMethod")
@Composable
private fun EnvironmentMetrics(
@ -541,29 +537,6 @@ private fun EnvironmentMetrics(
}
}
@Suppress("MagicNumber")
private fun Float.toTempString(isFahrenheit: Boolean) = if (isFahrenheit) {
val fahrenheit = this * 1.8F + 32
"%.0f°F".format(fahrenheit)
} else {
"%.0f°C".format(this)
}
@Suppress("MagicNumber")
private fun Float.toSpeedString() = when (DistanceUnit.getFromLocale()) {
DisplayUnits.METRIC -> "%.0f km/h".format(this * 3.6)
else -> "%.0f mph".format(this * 2.23694f)
}
// Magnus-Tetens approximation
@Suppress("MagicNumber")
private fun calculateDewPoint(tempCelsius: Float, humidity: Float): Float {
val (a, b) = 17.27f to 237.7f
val alpha = (a * tempCelsius) / (b + tempCelsius) + ln(humidity / 100f)
return (b * alpha) / (a - alpha)
}
@OptIn(ExperimentalLayoutApi::class)
@Composable
private fun PowerMetrics(node: Node) = with(node.powerMetrics) {
FlowRow(

View file

@ -66,6 +66,7 @@ import com.geeksville.mesh.ui.components.CommonCharts.MS_PER_SEC
import com.geeksville.mesh.ui.components.CommonCharts.DATE_TIME_FORMAT
import com.geeksville.mesh.util.GraphUtil.createPath
import com.geeksville.mesh.util.GraphUtil.drawPathWithGradient
import com.geeksville.mesh.util.UnitConversions.celsiusToFahrenheit
private val LEGEND_DATA_1 = listOf(
LegendData(
@ -102,12 +103,6 @@ fun EnvironmentMetricsScreen(
val graphData = environmentState.environmentMetricsFiltered(selectedTimeFrame)
val data = graphData.metrics
/* Convert Celsius to Fahrenheit */
@Suppress("MagicNumber")
fun celsiusToFahrenheit(celsius: Float): Float {
return (celsius * 1.8F) + 32
}
val processedTelemetries: List<Telemetry> = if (state.isFahrenheit) {
data.map { telemetry ->
val temperatureFahrenheit =

View file

@ -25,7 +25,6 @@ import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.BoxWithConstraints
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.ColumnScope
import androidx.compose.foundation.layout.ExperimentalLayoutApi
import androidx.compose.foundation.layout.FlowRow
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.RowScope
@ -98,15 +97,15 @@ private fun HeaderItem(compactWidth: Boolean) {
.padding(8.dp),
horizontalArrangement = Arrangement.SpaceBetween,
) {
PositionText("Latitude", Weight20)
PositionText("Longitude", Weight20)
PositionText("Sats", Weight10)
PositionText("Alt", Weight15)
PositionText(stringResource(R.string.latitude), Weight20)
PositionText(stringResource(R.string.longitude), Weight20)
PositionText(stringResource(R.string.sats), Weight10)
PositionText(stringResource(R.string.alt), Weight15)
if (!compactWidth) {
PositionText("Speed", Weight15)
PositionText("Heading", Weight15)
PositionText(stringResource(R.string.heading), Weight15)
}
PositionText("Timestamp", Weight40)
PositionText(stringResource(R.string.timestamp), Weight40)
}
}
@ -155,7 +154,6 @@ private fun formatPositionTime(
return timeText
}
@OptIn(ExperimentalLayoutApi::class)
@Composable
private fun ActionButtons(
clearButtonEnabled: Boolean,

View file

@ -19,7 +19,6 @@ package com.geeksville.mesh.ui.radioconfig.components
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.ExperimentalLayoutApi
import androidx.compose.foundation.layout.FlowRow
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.fillMaxWidth
@ -55,7 +54,6 @@ import com.geeksville.mesh.ui.components.PositionPrecisionPreference
import com.geeksville.mesh.ui.components.SwitchPreference
@Suppress("LongMethod")
@OptIn(ExperimentalLayoutApi::class)
@Composable
fun EditChannelDialog(
channelSettings: ChannelProtos.ChannelSettings,

View file

@ -19,7 +19,7 @@ package com.geeksville.mesh.util
import android.icu.util.LocaleData
import android.icu.util.ULocale
import com.geeksville.mesh.ConfigProtos.Config.DisplayConfig
import com.geeksville.mesh.ConfigProtos.Config.DisplayConfig.DisplayUnits
import java.util.Locale
enum class DistanceUnit(
@ -27,23 +27,23 @@ enum class DistanceUnit(
val multiplier: Float,
val system: Int
) {
METER("m", multiplier = 1F, DisplayConfig.DisplayUnits.METRIC_VALUE),
KILOMETER("km", multiplier = 0.001F, DisplayConfig.DisplayUnits.METRIC_VALUE),
FOOT("ft", multiplier = 3.28084F, DisplayConfig.DisplayUnits.IMPERIAL_VALUE),
MILE("mi", multiplier = 0.000621371F, DisplayConfig.DisplayUnits.IMPERIAL_VALUE),
METER("m", multiplier = 1F, DisplayUnits.METRIC_VALUE),
KILOMETER("km", multiplier = 0.001F, DisplayUnits.METRIC_VALUE),
FOOT("ft", multiplier = 3.28084F, DisplayUnits.IMPERIAL_VALUE),
MILE("mi", multiplier = 0.000621371F, DisplayUnits.IMPERIAL_VALUE),
;
companion object {
fun getFromLocale(locale: Locale = Locale.getDefault()): DisplayConfig.DisplayUnits {
fun getFromLocale(locale: Locale = Locale.getDefault()): DisplayUnits {
return if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.P) {
when (LocaleData.getMeasurementSystem(ULocale.forLocale(locale))) {
LocaleData.MeasurementSystem.SI -> DisplayConfig.DisplayUnits.METRIC
else -> DisplayConfig.DisplayUnits.IMPERIAL
LocaleData.MeasurementSystem.SI -> DisplayUnits.METRIC
else -> DisplayUnits.IMPERIAL
}
} else {
when (locale.country.uppercase(locale)) {
"US", "LR", "MM", "GB" -> DisplayConfig.DisplayUnits.IMPERIAL
else -> DisplayConfig.DisplayUnits.METRIC
"US", "LR", "MM", "GB" -> DisplayUnits.IMPERIAL
else -> DisplayUnits.METRIC
}
}
}
@ -54,9 +54,9 @@ fun Int.metersIn(unit: DistanceUnit): Float {
return this * unit.multiplier
}
fun Int.metersIn(system: DisplayConfig.DisplayUnits): Float {
fun Int.metersIn(system: DisplayUnits): Float {
val unit = when (system.number) {
DisplayConfig.DisplayUnits.IMPERIAL_VALUE -> DistanceUnit.FOOT
DisplayUnits.IMPERIAL_VALUE -> DistanceUnit.FOOT
else -> DistanceUnit.METER
}
return this.metersIn(unit)
@ -70,9 +70,9 @@ fun Float.toString(unit: DistanceUnit): String {
}.format(this, unit.symbol)
}
fun Float.toString(system: DisplayConfig.DisplayUnits): String {
fun Float.toString(system: DisplayUnits): String {
val unit = when (system.number) {
DisplayConfig.DisplayUnits.IMPERIAL_VALUE -> DistanceUnit.FOOT
DisplayUnits.IMPERIAL_VALUE -> DistanceUnit.FOOT
else -> DistanceUnit.METER
}
return this.toString(unit)
@ -80,8 +80,8 @@ fun Float.toString(system: DisplayConfig.DisplayUnits): String {
private const val KILOMETER_THRESHOLD = 1000
private const val MILE_THRESHOLD = 1609
fun Int.toDistanceString(system: DisplayConfig.DisplayUnits): String {
val unit = if (system.number == DisplayConfig.DisplayUnits.METRIC_VALUE) {
fun Int.toDistanceString(system: DisplayUnits): String {
val unit = if (system.number == DisplayUnits.METRIC_VALUE) {
if (this < KILOMETER_THRESHOLD) DistanceUnit.METER else DistanceUnit.KILOMETER
} else {
if (this < MILE_THRESHOLD) DistanceUnit.FOOT else DistanceUnit.MILE
@ -89,3 +89,9 @@ fun Int.toDistanceString(system: DisplayConfig.DisplayUnits): String {
val valueInUnit = this * unit.multiplier
return valueInUnit.toString(unit)
}
@Suppress("MagicNumber")
fun Float.toSpeedString() = when (DistanceUnit.getFromLocale()) {
DisplayUnits.METRIC -> "%.0f km/h".format(this * 3.6)
else -> "%.0f mph".format(this * 2.23694f)
}

View file

@ -0,0 +1,46 @@
/*
* 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.util
import kotlin.math.ln
object UnitConversions {
@Suppress("MagicNumber")
fun celsiusToFahrenheit(celsius: Float): Float {
return (celsius * 1.8F) + 32
}
fun Float.toTempString(isFahrenheit: Boolean) = if (isFahrenheit) {
val fahrenheit = celsiusToFahrenheit(this)
"%.0f°F".format(fahrenheit)
} else {
"%.0f°C".format(this)
}
/**
* Calculated the dew point based on the Magnus-Tetens approximation which is a widely used
* formula for calculating dew point temperature.
*/
@Suppress("MagicNumber")
fun calculateDewPoint(tempCelsius: Float, humidity: Float): Float {
val (a, b) = 17.27f to 237.7f
val alpha = (a * tempCelsius) / (b + tempCelsius) + ln(humidity / 100f)
return (b * alpha) / (a - alpha)
}
}

View file

@ -595,4 +595,14 @@
<string name="backup_restore"><![CDATA[Backup & Restore]]></string>
<string name="import_configuration">Import configuration</string>
<string name="export_configuration">Export configuration</string>
<string name="hardware">Hardware</string>
<string name="supported">Supported</string>
<string name="node_number">Node Number</string>
<string name="user_id">User ID</string>
<string name="uptime">Uptime</string>
<string name="firmware_version">Firmware version</string>
<string name="timestamp">Timestamp</string>
<string name="heading">Heading</string>
<string name="sats">Sats</string>
<string name="alt">Alt</string>
</resources>