Status Update
Comments
ma...@google.com <ma...@google.com>
an...@google.com <an...@google.com>
da...@gmail.com <da...@gmail.com> #2
After upgrading to kotlin 2.1.0 and bumping remaining dependencies, things seem to work as expected again. EDIT: My bad; was on the wrong branch. Those upgrades did NOT help.
ch...@google.com <ch...@google.com> #3
I think I managed to track it down further:
In my Child
composable, I'm actually hitting a TODO()
statement:
@Composable
fun Child(isLoading: Boolean, items: List<MyDataClass>) {
// ...
if (notYetSupported) {
TODO("This is not yet supported!")
}
// ...
}
It would seem that compose swallows this error and throws the above instead which is not very helpful.
ae...@gmail.com <ae...@gmail.com> #4
The "pending composition has not been applied" error generally means that composition continued despite something in the hierarchy previously throwing an exception. Here it looks like something is continuing a subcomposition unexpectedly, possibly a LazyList.
We should see if we can improve this error messaging. Can you send over a more complete example? The snippets here aren't fleshed out enough to reproduce this swallowing behavior.
ol...@gmail.com <ol...@gmail.com> #5
Triage notes: It feels like we're very close to having a repro case that we could run internally, but we don't know where the sub-composition is coming from.
Can you send over a more complete example?
Let's hold until we can get a stable repro case from the reporter. Could you send us a ZIP of the project that repros this issue? We'll close this out in a month if we don't get a response.
Bugjuggler: wait 1 month
ae...@gmail.com <ae...@gmail.com> #6
an...@gmail.com <an...@gmail.com> #7
This took me about an hour to extract but it reproduces the issue for me.
@file:SuppressLint("UnusedMaterialScaffoldPaddingParameter")
package com.example
import android.annotation.SuppressLint
import android.app.Application
import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.animation.AnimatedVisibility
import androidx.compose.material.MaterialTheme
import androidx.compose.material.Scaffold
import androidx.compose.material.Text
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import kotlinx.coroutines.delay
class MyApp : Application()
class MainActivity : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
MaterialTheme {
// Scaffold is required!
Scaffold { _ ->
val a = remember { mutableStateOf(true) }
val b = remember { mutableStateOf(true) }
LaunchedEffect(Unit) {
a.value = false
delay(50)
b.value = false
}
AnimatedVisibility(visible = a.value) { Text("Hello World") }
if (!b.value) {
TODO()
}
}
}
}
}
}
jo...@gmail.com <jo...@gmail.com> #8
Thank you! I was able to reproduce locally. Will look into this. We may eventually dedupe to
One thing to note is that this is definitely related to a race condition. Device speed matters, and a slower device seems more likely to trigger this exception — My aging Pixel XL from 2016 reproduces this very consistently, unlike my local emulator which doesn't demonstrate this issue at all. We had suspected that coroutines were coming into play here, so a lot of pieces are starting to come together.
ae...@gmail.com <ae...@gmail.com> #9
Correction: The emulator I was using had animations disabled. This seems to reproduce very consistently across devices.
It looks like our exception handling logic is try/catching Exception
instead of Throwable
. TODO() is breaking things because it throws a NotImplementedError
, which isn't caught by our error handler and doesn't cancel pending recompositions. We'll update our code to include Errors and generic Throwable classes. It's not immediately obvious if this will fix other occurrences of this exception since we don't have other reproduction cases.
ol...@gmail.com <ol...@gmail.com> #10
Project: platform/frameworks/support
Branch: androidx-main
Author: Andrew Bailey <
Link:
Fix Throwables causing unapplied composition error
Expand for full commit details
Fix Throwables causing unapplied composition error
The error handling logic in the Recomposer was set up to catch
throwables that extended from Exception, which excluded Errors (like
NotImplementedError, error(""), StackOverflowError, etc.) and
user-defined exceptions that extend directly from Throwable. This CL
updates the relevant pathways to switch over all Throwables.
This resolves at least one pathway that leads to a "Pending composition
has not been applied." runtime error. It's not immediately clear if this
accounts for all of the reports since we only have one known
reproduction case right now.
Test: Manually run repro case in the linked bug
Fixes: b/382094412
Relnote: """
Fixes an issue where raising a throwable during composition that
does not extend from Exception may lead to a 'Pending composition
has not been applied' error.
"""
Change-Id: I356be5d99df41138be790275807544b2d717050c
Files:
- M
compose/runtime/runtime/api/current.txt
- M
compose/runtime/runtime/api/restricted_current.txt
- M
compose/runtime/runtime/integration-tests/src/androidInstrumentedTest/kotlin/androidx/compose/runtime/LiveEditApiTests.kt
- M
compose/runtime/runtime/src/commonMain/kotlin/androidx/compose/runtime/HotReloader.kt
- M
compose/runtime/runtime/src/commonMain/kotlin/androidx/compose/runtime/Recomposer.kt
- M
compose/runtime/runtime/src/nonEmulatorJvmTest/kotlin/androidx/compose/runtime/LiveEditTests.kt
Hash: 120c3f0cfa04436bd9afabea415a478e12458316
Date: Wed Jan 08 15:38:34 2025
ae...@gmail.com <ae...@gmail.com> #11
Please see the duplicated issue:
as...@gmail.com <as...@gmail.com> #12
Please treat this as a critical bug.
Fatal Exception: java.lang.IllegalStateException: You cannot access the NavBackStackEntry's ViewModels after the NavBackStackEntry is destroyed.
at androidx.navigation.NavBackStackEntry.getViewModelStore(NavBackStackEntry.kt:202)
at androidx.navigation.compose.NavHostKt.NavHost(NavHost.kt:104)
at androidx.navigation.compose.NavHostKt.NavHost(NavHost.kt:67)
ch...@google.com <ch...@google.com> #13
Compose does not ensure repeatable read consistency across a sub-composition boundary (that is we do not guarantee that a mutable value read by the parent composition will be the same value read by the sub-composition).
However, as I noted above, we will be changing the runtime to avoid recomposing sub-compositions that have been detected in the parent composition as being removed, point (2) in
These two changes will change the rules so that the assumption how data dependencies are handled across sub-composition boundaries, as is implied by the assert
above in setContent
of a Composition
to be violated. When and why setContent
is called is in the hands to the code that created the composition, not the recomposer, or the rest of the runtime, which is why we cannot guarantee the assert assumption will always hold.
Theses changes will not ensure repeatable read consistency across a sub-composition boundary, but they will make it much easier for code to avoid requiring it by avoiding recomposing code that is being removed. Because repeatable read consistency is not guaranteed it is recommended that code not rely on it and avoid it by avoiding repeating reads as shown in the examples in
I am sorry if my calling this a feature request above implied a lack of priority. We are treating this change as a high priority change for the next feature release. As it requires an API change and a change to the rules of how composition works I think of it as a feature request but will leave it marked as a bug type to avoid confusion.
ap...@google.com <ap...@google.com> #14
Branch: androidx-main
commit 9f3a404d3117ed29b3ced1f2e29134237bf6baa1
Author: Ben Trengrove <bentrengrove@google.com>
Date: Thu Jun 01 15:56:39 2023
Skip recomposition of subcompositions that will be removed
Compose does not ensure repeatable read consistency across a sub-composition boundary (that is we do not guarantee that a mutable value read by the parent composition will be the same value read by the sub-composition). However in cases where we can detect that a subcomposition will be removed in the next applyChanges, we can skip recomposing that
subcomposition.
This is accomplished by keeping a map of removed compositions reported via the composition context.
When a composition is marked for deletion, it checks through its marked groups for any nested composition contexts. For each one it finds, reports its deletion to the recomposer via the context. When the nested composition is
recomposed, if the recomposers map of removed compositions contains the composition, it is skipped.
This change is an improvement on current behaviour but could be improved by ensuring parent compositions are always recomposed before any nested compositions. This mostly already occurs but making this enforced would further improve this situation and is why the linked bug is not yet considered fixed.
Bug: 254645321
Relnote: Skip recomposition of subcompositions that will be removed
Test: testSubcompositionDisposedInParent
Change-Id: Ieeb9919897a9f9b4274ddc77e66608a673cd1112
M compose/runtime/runtime/src/commonMain/kotlin/androidx/compose/runtime/Composer.kt
M compose/runtime/runtime/src/commonMain/kotlin/androidx/compose/runtime/CompositionContext.kt
M compose/runtime/runtime/src/commonMain/kotlin/androidx/compose/runtime/Recomposer.kt
M compose/runtime/runtime/src/nonEmulatorCommonTest/kotlin/androidx/compose/runtime/RecomposerTests.kt
ch...@google.com <ch...@google.com> #15
As Ben and I found a relatively fast and safe way to fix this that didn't involve an API changes, we have started the process to cherry-pick this to earlier releases.
ri...@ffw.com <ri...@ffw.com> #16
Which version is this in / going to be in?
ch...@google.com <ch...@google.com> #17
This issue will be updated when it lands in a stable release.
ae...@gmail.com <ae...@gmail.com> #18
ch...@google.com <ch...@google.com> #19
I could not reproduce the crash above with 1.6.0-alpha01
The steps I took where:
- Created a new android project using
Android Studio Giraffe | 2022.3.1 RC 1
- Replaced
Greeting
with the code in #1 (fixing the imports) - Ran the program
- Pressed the button (crash reproduced)
- Added the line
implementation("androidx.compose.runtime:runtime:1.6.0-alpha01")
to the dependencies - Update the compile sdk to 34.
- Sync'ed gradle
- Ran the program
- Pressed the button (no crash)
ch...@google.com <ch...@google.com> #20
Note the above fix is only a partial fix. There are cases that will continue to violate the implied constraint until the second part of the fix lands (explained above).
Also note that this doesn't guarantee the constraint implied by the assert will always hold, only that the runtime will try to avoid recomposing sub-compositions that might be exposed to the constraint. Any use of sub-composition can violate this constraint as the value might change between the time the main composition composes and the sub-composition composes and before the main composition can be recomposed to remove the sub-composition.
If you require consistency between a composition and its subcomposition you need to capture the state as #5, above. This change just avoids a common case would violate this constraint.
Description
Jetpack Compose version: 1.2.1 (also tested on 1.3.0-rc01)
Jetpack Compose component(s) used: foundation, material
Here is the most simplified version of the code that led me to mysterious crashes when using ModalBottomSheetLayout:
When the button is clicked the code crashes, although you would not expect this just looking at the code. You would expect BoxWithConstraints to be removed from composition as soon as the
value
becomesfalse
.And if you replace BoxWithConstraints with regular Box it would work without assertion crash however.
If I understand correctly, the SubcomposeLayout causes its content to be recomposed without regards to the outer check. Which may be fine if you know about this behaviour and expect it. However in case of higher level composables such as ModalBottomSheetLayout it hides the underlying behaviour which may lead to similar unexpected crashes.
I don't know but to me it seems like a flaw. Please correct me if I'm just not doing something correctly.