Why SideEffect gets called ever-time my composable is invalidated , but same does-not hold true for LaunchedEffect?
sealed class SomeState { object Error:SomeState() data class Content(): SomeState } class MyViewModel:ViewModel { internal val response: MutableLiveData<SomeState> by lazy { MutableLiveData<SomeState>() } } // This is top-level composable, it wont be recomposed ever @Composable fun MyComposableScreen( viewModel:MyVm, launchActivity:()->Unit ){ val someDialog = remember { mutableStateOf(false) } MyComposableContent() GenericErrorDialog(someDialog = someDialog) when (val state = viewModel.response.observeAsState().value) { // Query 1 is Content -> LaunchedEffect(Unit) { launchActivity() } Error -> { // Query 2 // Gets called everytime this composable gets invalidated, for eg in case of TextField change, compiler is invalidating it. // But if i change it to LaunchedEffect(Unit), invalidation has no effect,LaunchedEffect only gets called when there is new update to the LiveData. why? SideEffect { someDialog.value = true} } } } // This is the content, which can be recomposed in case of email is changed @Composable fun MyComposableContent( onEmailChange:(email) -> Unit, email:String, ){ TextField( email = email, onValueChange = onEmailChange ) } I have doubts related to Query 1 and Query 2 both are part of top-level composable which will never be re-composed, but can be invalidated,
when (val state = viewModel.response.observeAsState().value) { // observing to live-data // Query 1 is Content -> LaunchedEffect(Unit) { launchActivity() } Error -> { // Query 2 SideEffect { someDialog.value = true} } } In case of is
Content -> LaunchedEffect(Unit) { launchActivity() } I believe this should be fine as we want to launch an activity only when LaunchedEffect is part of the first time composition, and it will be only part of the composition if live data state is Content
I faced issue in second scenario,
Error -> { // Query 2 SideEffect { someDialog.value = true // shows a dialog} } If last state of live-data, is Error in viewModel. And every time i make changes in the TextField my top level MyComposableScreen was getting invalidated(not recomposed) by compose compiler, and since last state of live-data was set as error, SideEffect was running every time, which is fine as it should run for every successful composition and re-composition.
But, if i change it from SideEffect to LaunchedEffect(Unit){someDialog.value = true} dialog box was not showing up every time MyComposableScreen was invalidated, thats the desired behavior.
LaunchedEffect(Unit) gets called only if there live-data emits the new state again because of any UI-action. But, I am not sure regarding the reasoning behind it, why the code inside LaunchedEffect(Unit){someDialog.value = true} does not trigger after composable gets invalidated but the code inside SideEffect gets triggered after composable gets invalidated?
To make it more clear
I understand the difference
SideEffect -> on every successful composition and re-composition, if it's part of it LaunchedEffect -> when its enters composition and span across re-composition unless the keys are changed.
But in above scenario - this code particularly
@Composable fun MyTopLevelComposable(viewModel:MyViewModel){ when (val state = viewModel.response.observeAsState().value) { // observing live-data state is Content -> LaunchedEffect(Unit) { launchActivity() } Error -> SideEffect { someDialog.value = true} } } It will never get recomposed. The only reason for this composable to be called again could be if compose compiler invalidates the view.
My Query is -> when view/composable gets invalidated
SideEffect {someDialog.value = true} executes, because it will again go through composition not re-composition as viewModel.response(which is live-data) last state was Error
But if change it to LaunchedEffect(Unit) {someDialog.value = true} it doesn't executes again after the composable is invalidated. It only reacts to a new state emitted by the live-data.
Question is why? Invalidate should start composition again, and since it's a composition. not re-composition LaunchedEffect should behave similarly to SideEffect in this scenario, as both are reacting to composition.