George
08/05/2022, 7:17 PMGeorge
08/05/2022, 7:17 PM@Composable
fun MyComposable(viewModel: MyViewModel = viewModel()) {
    Text(text = "${viewModel.loading}")
}
class MyViewModel : ViewModel() {
    var loading by mutableStateOf(false)
    init {
        viewModelScope.launch {
            loading = true
            SomeRepo.loadSomeData()
            loading = false
        }
    }
}<http://Dispatchers.IO|Dispatchers.IO>"IllegalStateException: Reading a state that was created after the snapshot was taken or in a snapshot that has not yet been applied"SavedStateHandleLaunchedEffectfun initialize() {
  if (initialized) return
  ...
  initialized = true
}LaunchedEffectinitMutableStateMutableStatedewildte
08/05/2022, 7:43 PMdewildte
08/05/2022, 7:49 PMLaunchedEffect(key1 = Unit) {
  viewModel.load()
}Zach Klippenstein (he/him) [MOD]
08/05/2022, 8:02 PMSnapshot.withMutableSnapshotBryan Herbst
08/05/2022, 8:03 PMLaunchedEffectinitZach Klippenstein (he/him) [MOD]
08/05/2022, 8:05 PMGeorge
08/05/2022, 9:27 PMSnapshot.withMutableSnapshotGeorge
08/05/2022, 9:36 PMinitGeorge
08/05/2022, 9:39 PMTry keeping Compose out of the ViewModel by using (State)Flows instead.Reverting to `StateFlow`s is definitely possible, it's just more boilerplate and more work for runtime to convert one reactive system to another.
Alex Vanyo
08/05/2022, 9:58 PMI think it’s better than it being a side effect of some ephemeral Composable.In some sense, that’s what the
initViewModelViewModelViewModelinitLaunchedEffectDisposableEffectGeorge
08/05/2022, 9:58 PMSimply instantiating an object shouldn’t trigger I/O operationsConstructor itself shouldn't perform (synchronous) I/O, sure. But the app needs to load data as soon as it is started, or a specific screen is reached. In this case starting the operation upon creation of a Repository or a ViewModel looks to me like a most clear & direct representation of this requirement in code. For example, when we're calling
Room.databaseBuilder(...).build()George
08/05/2022, 10:01 PMGeorge
08/05/2022, 10:06 PMGeorge
08/05/2022, 10:12 PMThe solution of moving it into a LaunchedEffectalso has a downside that the first composition will have an inconsistent state to display: not loading and no data. Or we would need to initialize
loadingtruedewildte
08/05/2022, 11:05 PMfun MyComposable(viewModel: MyViewModel = viewModel())dewildte
08/05/2022, 11:14 PMthe first composition will have an inconsistent state to display: not loading and no data.This is called an EMPTY state and is a perfectly valid state to be in. And you could even have the default for
loadingtrueGeorge
08/06/2022, 12:05 AMThe bottom line is that the solutions proposed to you workSure. I've mentioned all of these options in my first post and explained their downsides (from my point of view). And I'm not going to convince anybody that this is a best possible architecture. What I'm trying to say is: this is a valid use case too, which worked reliably with Fragments but may have an unexpected gotcha in pure Compose. If this could be solved technically, once and for all, it would be the best possible outcome for everyone.
This is called an EMPTY state and is a perfectly valid state to be inEmpty data may be fine ("your basket is empty") or represent an error ("order not found"), but in either case I wouldn't show that for a split second before we had a chance to actually load the data. I'd call it an "inconsistent" UI, sorry if I wasn't clear enough.
Zach Klippenstein (he/him) [MOD]
08/06/2022, 1:30 AMviewModelScopeDispatchers.Main.immediateDispatchers.MainMainMain.immediateZach Klippenstein (he/him) [MOD]
08/06/2022, 1:30 AMvar loading by Snapshot.withMutableSnapshot { mutableStateOf(false) }CoroutineStart.UNDISPATCHEDsuspend fun awaitGlobalSnapshotReady() {
    suspendCancellableCoroutine { continuation ->
        lateinit var handle: ObserverHandle
        handle = Snapshot.registerApplyObserver { _, _ ->
            handle.dispose()
            continuation.resume(Unit)
        }
        continuation.invokeOnCancellation {
            handle.dispose()
        }
    }
}George
08/06/2022, 8:27 AMPut both the state initialization and the write in explicit snapshotsWow, that works, but honestly I can't understand why. When a nested snapshot is taken and immediately applied, what difference does it make?
Make your coroutine explicitly wait for the next global snapshot applicationEven if it's not practical for this use case, it's exciting to know that such a thing is possible. But there is no way to "bubble up" VM initialization to global snapshot or to apply current snapshot prematurely, right? (although I'm not sure if the latter, called immediately after VM creation, would solve the issue: it might be too late)
Zach Klippenstein (he/him) [MOD]
08/06/2022, 3:20 PMZach Klippenstein (he/him) [MOD]
08/06/2022, 3:25 PMviewModelGeorge
08/06/2022, 5:38 PMif the composition is abandoned, any VMs it's caused to instantiate won't be disposed, which means that any side effects performed in the VM constructor (such as coroutines launched in the vmScope) won't be disposed and will just be leaked by being allowed to continue to run indefinitelyI would say it's a feature, not a bug. Next composition will get from
ViewModelStoreOwnerGeorge
08/06/2022, 6:47 PMYou could break out of the current snapshot and initialize the VM in the global snapshot explicitly, but then it wouldn't be available in the composition snapshot at allThanks for the clarification, I haven't realized this.
it would have to handle the case where the snapshot is disposed without applicationJust curious: what may cause composition to fail in a typical Android app? How often does that happen?
Zach Klippenstein (he/him) [MOD]
08/07/2022, 5:53 AMZach Klippenstein (he/him) [MOD]
08/07/2022, 6:00 AMGeorge
08/07/2022, 11:57 AM@Composable
fun MyComposable(viewModel: MyViewModel = viewModel()) {
    LaunchedEffect(Unit) {
        viewModel.initialize()
    }
    Text(text = "${viewModel.loading}")
}
class MyViewModel : ViewModel() {
    // `loading` should be initialized to `true` to prevent showing wrong state on first composition, before LaunchedEffect kicks in
    var loading by mutableStateOf(true)
    private var initialized = AtomicBoolean(false)
    fun initialize() {
        if (!initialized.compareAndSet(false, true)) {
            return
        }
        viewModelScope.launch(<http://Dispatchers.IO|Dispatchers.IO>) {
            loading = true
            SomeRepo.loadSomeData()
            loading = false
        }
    }
}loadingtrueloadSomeDataZach Klippenstein (he/him) [MOD]
08/07/2022, 2:57 PM