Há muitas perguntas semelhantes sobre isso, mas os usos estão mudando; esta é a única pergunta que encontrei que fala especificamente sobre MutableSets, MutableStateFlows e collectAsState().
O problema é que quando adiciono um elemento ao meu MutableSet, o composable não recompõe. O código a seguir é o mais simples que posso obter e ainda mostra o problema.
Modelo de visualização
class MyViewmodel : ViewModel() {
private val _infoSet = MutableStateFlow(mutableSetOf<MyInfo>())
val infoSet = _infoSet.asStateFlow()
fun add() {
val newInfo = MyInfo(num = infoSet.value.size + 1)
_infoSet.value.add(newInfo)
Log.d(TAG, "infoSet is now ${infoSet.value}")
}
}
data class MyInfo(
val name: String = "foo",
val num: Int = -1
)
Atividade principal
class MainActivity : ComponentActivity() {
private lateinit var myViewmodel: MyViewmodel
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
enableEdgeToEdge()
myViewmodel = MyViewmodel()
setContent {
val infoSet by myViewmodel.infoSet.collectAsStateWithLifecycle()
RecompositionTestTheme {
Scaffold(modifier = Modifier.fillMaxSize()) { innerPadding ->
DrawStuff(
modifier = Modifier.padding(innerPadding),
infoSet = infoSet,
myViewmodel
)
}
}
}
}
}
@Composable
fun DrawStuff(
modifier: Modifier = Modifier,
infoSet: Set<MyInfo>,
viewmodel: MyViewmodel
) {
Log.d(TAG, "DrawStuff() start")
LazyColumn(modifier = modifier.safeDrawingPadding()) {
item { Text("This should display a Set of MyInfo") }
item {
Button(
onClick = {
viewmodel.add()
Toast.makeText(ctx, "infoSet = $infoSet", Toast.LENGTH_SHORT).show()
}
) { Text("add") }
}
// the MyInfo list
infoSet.forEach { info ->
Log.d(TAG, "listing the MyInfos $info")
item {
Text("${info.name}, ${info.num}")
}
}
}
}
As instruções de Log mostram que clicar no botão chama MyViewmodel.add(). Mas os logs também mostram claramente que DrawStuff não é chamado. Inesperadamente, a mensagem Toast mostra corretamente as informações.
Há algo que estou fazendo errado aqui? O que preciso fazer para exibir DrawStuff()
o conteúdo de infoSet
quando ele muda? (Nota: Fiz testes onde infoSet foi pré-carregado com dados. Ele desenha corretamente, mas ainda não muda quando itens são adicionados.)