¿Por qué la vista sigue parpadeando cuando se usa la navegación jetpack con Compose?

2 minutos de lectura

avatar de usuario
androider2

Tengo una pantalla de inicio de sesión y cuando el inicio de sesión es exitoso y el modelo de vista actualiza la variable de estado mutable, mi expectativa es que se llame a una nueva función componible para mostrar una nueva pantalla y se elimine la de inicio de sesión. El problema es que cuando la nueva pantalla (aka Screen.AccountsScreen), su contenido sigue parpadeando/redibujando y sucede lo mismo con el formulario de inicio de sesión que nunca se destruye (lo sé porque el mensaje de registro ‘Recomponing…’ se imprime sin fin). Supongo que esto sucede porque el isLoginSuccessful el estado es siempre verdadero. Parece que necesito un evento que solo se pueda consumir una vez, ¿es correcto? Si es así, ¿cómo puedo hacer eso?

LoginViewModel.kt

@HiltViewModel
class LoginViewModel @Inject constructor() : ViewModel() {

  var isLoginSuccessful by mutableStateOf(false)
  var errorMessage by mutableStateOf("")
  
  fun onLoginClick(email: String, password:String) {
    errorMessage = ""
    if (credentialsValid(email, password)) {
      isLoginSuccessful = true
    } else {
      errorMessage = "Email or password invalid"
      isLoginSuccessful = false
    }
  }
}

LoginScreen.kt

@Composable
fun loginScreen(
  navController: NavController,
  viewModel: LoginViewModel = hiltViewModel()
) {
  println("Recomponing...")
  // Here gos the code for the login form
  
  if (viewModel.isLoginSuccessful) {
    navController.navigate(Screen.AccountsScreen.route) {
      popUpTo(Screen.LoginScreen.route) { inclusive = true }
    }
  }
}

  • ¡Hola! ¿Mi respuesta resolvió tu pregunta? Si es así, acéptelo con una marca de verificación debajo del contador de votos. De lo contrario, avísame si tienes algún problema con él.

    – Pylyp Dujov

    18 oct 2021 a las 16:11

avatar de usuario
Pylyp Dujov

La navegación compuesta recompone las vistas que aparecen y desaparecen durante la transición. Este es el comportamiento esperado.

Estás llamando a navegar en cada recomposición. Tu problema radica en estas líneas:

if (viewModel.isLoginSuccessful) {
    navController.navigate(Screen.AccountsScreen.route) {
        popUpTo(Screen.LoginScreen.route) { inclusive = true }
    }
}

No debe cambiar el estado directamente desde los constructores de vistas. En este caso LaunchedEffect debería ser usado:

if (viewModel.isLoginSuccessful) {
    LaunchedEffect(Unit) {
        navController.navigate(Screen.AccountsScreen.route) {
            popUpTo(Screen.LoginScreen.route) { inclusive = true }
        }
    }
}

Vea más en efectos secundarios documentación.

¿Ha sido útil esta solución?

Esta web utiliza cookies propias y de terceros para su correcto funcionamiento y para fines analíticos y para mostrarte publicidad relacionada con sus preferencias en base a un perfil elaborado a partir de tus hábitos de navegación. Al hacer clic en el botón Aceptar, acepta el uso de estas tecnologías y el procesamiento de tus datos para estos propósitos. Configurar y más información
Privacidad