mirror of
https://github.com/meshtastic/Meshtastic-Android.git
synced 2026-04-20 22:23:37 +00:00
Fix: Periodically update last heard time (#1178)
This commit is contained in:
parent
acbae6d93d
commit
1f05886873
4 changed files with 66 additions and 21 deletions
|
|
@ -7,6 +7,9 @@ import androidx.compose.material.Icon
|
|||
import androidx.compose.material.MaterialTheme
|
||||
import androidx.compose.material.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.getValue
|
||||
import androidx.compose.runtime.mutableStateOf
|
||||
import androidx.compose.runtime.remember
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.graphics.vector.ImageVector
|
||||
|
|
@ -20,8 +23,10 @@ import com.geeksville.mesh.util.formatAgo
|
|||
@Composable
|
||||
fun LastHeardInfo(
|
||||
modifier: Modifier = Modifier,
|
||||
lastHeard: Int
|
||||
lastHeard: Int,
|
||||
currentTimeMillis: Long,
|
||||
) {
|
||||
|
||||
Row(
|
||||
modifier = modifier,
|
||||
verticalAlignment = Alignment.CenterVertically,
|
||||
|
|
@ -34,7 +39,7 @@ fun LastHeardInfo(
|
|||
tint = MaterialTheme.colors.onSurface,
|
||||
)
|
||||
Text(
|
||||
text = formatAgo(lastHeard),
|
||||
text = formatAgo(lastHeard, currentTimeMillis),
|
||||
color = MaterialTheme.colors.onSurface,
|
||||
fontSize = MaterialTheme.typography.button.fontSize
|
||||
)
|
||||
|
|
@ -45,6 +50,9 @@ fun LastHeardInfo(
|
|||
@Composable
|
||||
fun LastHeardInfoPreview() {
|
||||
AppTheme {
|
||||
LastHeardInfo(lastHeard = (System.currentTimeMillis() / 1000).toInt() - 8600)
|
||||
LastHeardInfo(
|
||||
lastHeard = (System.currentTimeMillis() / 1000).toInt() - 8600,
|
||||
currentTimeMillis = System.currentTimeMillis()
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -73,6 +73,7 @@ fun NodeInfo(
|
|||
chipClicked: () -> Unit = {},
|
||||
blinking: Boolean = false,
|
||||
expanded: Boolean = false,
|
||||
currentTimeMillis: Long,
|
||||
) {
|
||||
val unknownShortName = stringResource(id = R.string.unknown_node_short_name)
|
||||
val unknownLongName = stringResource(id = R.string.unknown_username)
|
||||
|
|
@ -157,7 +158,8 @@ fun NodeInfo(
|
|||
)
|
||||
|
||||
LastHeardInfo(
|
||||
lastHeard = thatNodeInfo.lastHeard
|
||||
lastHeard = thatNodeInfo.lastHeard,
|
||||
currentTimeMillis = currentTimeMillis
|
||||
)
|
||||
}
|
||||
Row(
|
||||
|
|
@ -273,7 +275,8 @@ fun NodeInfoSimplePreview() {
|
|||
thatNodeInfo = thatNodeInfo,
|
||||
1,
|
||||
0,
|
||||
true
|
||||
true,
|
||||
currentTimeMillis = System.currentTimeMillis()
|
||||
)
|
||||
}
|
||||
}
|
||||
|
|
@ -300,7 +303,8 @@ fun NodeInfoPreview(
|
|||
gpsFormat = 0,
|
||||
distanceUnits = 1,
|
||||
tempInFahrenheit = true,
|
||||
expanded = false
|
||||
expanded = false,
|
||||
currentTimeMillis = System.currentTimeMillis()
|
||||
)
|
||||
Text(
|
||||
text = "Details Shown",
|
||||
|
|
@ -312,7 +316,8 @@ fun NodeInfoPreview(
|
|||
gpsFormat = 0,
|
||||
distanceUnits = 1,
|
||||
tempInFahrenheit = true,
|
||||
expanded = true
|
||||
expanded = true,
|
||||
currentTimeMillis = System.currentTimeMillis()
|
||||
)
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,6 +1,7 @@
|
|||
package com.geeksville.mesh.ui
|
||||
|
||||
import android.os.Bundle
|
||||
import android.util.Log
|
||||
import android.view.LayoutInflater
|
||||
import android.view.View
|
||||
import android.view.ViewGroup
|
||||
|
|
@ -17,12 +18,17 @@ import androidx.compose.material.MaterialTheme
|
|||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.LaunchedEffect
|
||||
import androidx.compose.runtime.getValue
|
||||
import androidx.compose.runtime.mutableLongStateOf
|
||||
import androidx.compose.runtime.mutableStateOf
|
||||
import androidx.compose.runtime.remember
|
||||
import androidx.compose.runtime.setValue
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.platform.ComposeView
|
||||
import androidx.compose.ui.platform.ViewCompositionStrategy
|
||||
import androidx.compose.ui.unit.dp
|
||||
import androidx.fragment.app.activityViewModels
|
||||
import androidx.hilt.navigation.compose.hiltViewModel
|
||||
import androidx.lifecycle.compose.LifecycleResumeEffect
|
||||
import androidx.lifecycle.compose.collectAsStateWithLifecycle
|
||||
import com.geeksville.mesh.NodeInfo
|
||||
import com.geeksville.mesh.R
|
||||
|
|
@ -31,13 +37,15 @@ import com.geeksville.mesh.model.UIViewModel
|
|||
import com.geeksville.mesh.ui.components.NodeFilterTextField
|
||||
import com.geeksville.mesh.ui.theme.AppTheme
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.delay
|
||||
import kotlin.time.Duration.Companion.minutes
|
||||
|
||||
@AndroidEntryPoint
|
||||
class UsersFragment : ScreenFragment("Users"), Logging {
|
||||
|
||||
private val model: UIViewModel by activityViewModels()
|
||||
|
||||
private fun popup(node: NodeInfo) {
|
||||
private fun popup(node: NodeInfo) {
|
||||
if (!model.isConnected()) return
|
||||
val isOurNode = node.num == model.myNodeNum
|
||||
val ignoreIncomingList = model.ignoreIncomingList
|
||||
|
|
@ -136,6 +144,8 @@ fun NodesScreen(
|
|||
}
|
||||
}
|
||||
|
||||
val currentTimeMillis = getCurrentTimeMillisWithLifecycle()
|
||||
|
||||
LazyColumn(
|
||||
state = listState,
|
||||
modifier = Modifier.fillMaxSize(),
|
||||
|
|
@ -173,8 +183,32 @@ fun NodesScreen(
|
|||
isIgnored = state.ignoreIncomingList.contains(node.num),
|
||||
chipClicked = { chipClicked(node) },
|
||||
blinking = node == focusedNode,
|
||||
expanded = state.showDetails
|
||||
expanded = state.showDetails,
|
||||
currentTimeMillis = currentTimeMillis
|
||||
)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun getCurrentTimeMillisWithLifecycle(): Long {
|
||||
var currentTimeMillis by remember { mutableLongStateOf(System.currentTimeMillis()) }
|
||||
var running by remember { mutableStateOf(false) }
|
||||
LaunchedEffect(running) {
|
||||
if (running) {
|
||||
while (true) {
|
||||
delay(1.minutes)
|
||||
currentTimeMillis = System.currentTimeMillis()
|
||||
Log.d("NodesScreen", "NodesScreen: $currentTimeMillis")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
LifecycleResumeEffect(Unit) {
|
||||
running = true
|
||||
onPauseOrDispose {
|
||||
running = false
|
||||
}
|
||||
}
|
||||
return currentTimeMillis
|
||||
}
|
||||
|
|
|
|||
|
|
@ -31,18 +31,16 @@ fun Any.toPIIString() =
|
|||
|
||||
fun ByteArray.toHexString() = joinToString("") { "%02x".format(it) }
|
||||
|
||||
fun formatAgo(lastSeenUnix: Int): String {
|
||||
val currentTime = (System.currentTimeMillis() / 1000).toInt()
|
||||
fun formatAgo(lastSeenUnix: Int, currentTimeMillis: Long = System.currentTimeMillis()): String {
|
||||
val currentTime = (currentTimeMillis / 1000).toInt()
|
||||
val diffMin = (currentTime - lastSeenUnix) / 60
|
||||
if (diffMin < 1)
|
||||
return "now"
|
||||
if (diffMin < 60)
|
||||
return diffMin.toString() + " min"
|
||||
if (diffMin < 2880)
|
||||
return (diffMin / 60).toString() + " h"
|
||||
if (diffMin < 1440000)
|
||||
return (diffMin / (60 * 24)).toString() + " d"
|
||||
return "?"
|
||||
return when {
|
||||
diffMin < 1 -> "now"
|
||||
diffMin < 60 -> diffMin.toString() + " min"
|
||||
diffMin < 2880 -> (diffMin / 60).toString() + " h"
|
||||
diffMin < 1440000 -> (diffMin / (60 * 24)).toString() + " d"
|
||||
else -> "?"
|
||||
}
|
||||
}
|
||||
|
||||
/// Allows usage like email.onEditorAction(EditorInfo.IME_ACTION_NEXT, { confirm() })
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue