Fixed
Status Update
Comments
il...@google.com <il...@google.com> #3
Project: platform/frameworks/support
Branch: androidx-main
commit b4d5b1c0ff880fe411cb0820e2503f91c4c23440
Author: Jake Wharton <jw@squareup.com>
Date: Tue Mar 05 23:14:25 2024
Invert dependency between lifecycle-runtime-compose and compose-ui-ui
This breaks an effective dependency cycle between library groups and will enable the future use of lifecycle-runtime-compose in a multiplatform context without a Compose UI dependency.
Before:
lifecycle-runtime-compose --> compose-ui-ui --> [lifecycle-runtime & compose-runtime]
After:
compose-ui-ui --> lifecycle-runtime-compose --> [lifeycle-runtime & compose-runtime]
The definition of the LocalLifecycleOwner composition local previosly lived in compose-ui-ui, which is why lifecycle-runtime-compose took a dependency on it. That composition local provided a LifecycleOwner into the Compose UI-based composition, and lifecycle-runtime-compose provided APIs on top of that LifecycleOwner.
The entirety of lifecycle-runtime-compose's APIs were built on lifecyle-runtime and compose-runtime with the sole exception being the use of the composition local to obtain the LifecycleOwner. By defining the composition local directly within lifecyle-runtime-compose, the dependency can be inverted making compose-ui-ui depend on it (it already depended on the regular runtime). This automatically exposes the Compose-based lifecycle APIs to downstream users of Compose UI and opens up usage of those APIs in compositions other than that provided by Compose UI (e.g., Molecule).
Bug: 328263448
Test: ./gradlew -p compose/ui bOS
Test: ./gradlew -p lifecycle bOS
Relnote: `LocalLifecycleOwner` moved from Compose UI to lifecycle-runtime-compose so that its Compose-based helper APIs can be used outside of Compose UI.
Change-Id: I6c41b92eb6aaab67e7d733dfe3fe0b429b46becf
M camera/integration-tests/avsynctestapp/build.gradle
M compose/ui/ui/build.gradle
M compose/ui/ui/src/androidMain/kotlin/androidx/compose/ui/platform/AndroidCompositionLocals.android.kt
M lifecycle/lifecycle-runtime-compose/api/current.txt
M lifecycle/lifecycle-runtime-compose/api/restricted_current.txt
M lifecycle/lifecycle-runtime-compose/build.gradle
M lifecycle/lifecycle-runtime-compose/src/androidTest/java/androidx/lifecycle/compose/DropUnlessLifecycleTest.kt
M lifecycle/lifecycle-runtime-compose/src/androidTest/java/androidx/lifecycle/compose/LifecycleEffectTest.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/DropUnlessLifecycle.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/FlowExt.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/LifecycleEffect.kt
A lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/LocalLifecycleOwner.kt
M wear/compose/integration-tests/navigation/build.gradle
https://android-review.googlesource.com/2989145
Branch: androidx-main
commit b4d5b1c0ff880fe411cb0820e2503f91c4c23440
Author: Jake Wharton <jw@squareup.com>
Date: Tue Mar 05 23:14:25 2024
Invert dependency between lifecycle-runtime-compose and compose-ui-ui
This breaks an effective dependency cycle between library groups and will enable the future use of lifecycle-runtime-compose in a multiplatform context without a Compose UI dependency.
Before:
lifecycle-runtime-compose --> compose-ui-ui --> [lifecycle-runtime & compose-runtime]
After:
compose-ui-ui --> lifecycle-runtime-compose --> [lifeycle-runtime & compose-runtime]
The definition of the LocalLifecycleOwner composition local previosly lived in compose-ui-ui, which is why lifecycle-runtime-compose took a dependency on it. That composition local provided a LifecycleOwner into the Compose UI-based composition, and lifecycle-runtime-compose provided APIs on top of that LifecycleOwner.
The entirety of lifecycle-runtime-compose's APIs were built on lifecyle-runtime and compose-runtime with the sole exception being the use of the composition local to obtain the LifecycleOwner. By defining the composition local directly within lifecyle-runtime-compose, the dependency can be inverted making compose-ui-ui depend on it (it already depended on the regular runtime). This automatically exposes the Compose-based lifecycle APIs to downstream users of Compose UI and opens up usage of those APIs in compositions other than that provided by Compose UI (e.g., Molecule).
Bug: 328263448
Test: ./gradlew -p compose/ui bOS
Test: ./gradlew -p lifecycle bOS
Relnote: `LocalLifecycleOwner` moved from Compose UI to lifecycle-runtime-compose so that its Compose-based helper APIs can be used outside of Compose UI.
Change-Id: I6c41b92eb6aaab67e7d733dfe3fe0b429b46becf
M camera/integration-tests/avsynctestapp/build.gradle
M compose/ui/ui/build.gradle
M compose/ui/ui/src/androidMain/kotlin/androidx/compose/ui/platform/AndroidCompositionLocals.android.kt
M lifecycle/lifecycle-runtime-compose/api/current.txt
M lifecycle/lifecycle-runtime-compose/api/restricted_current.txt
M lifecycle/lifecycle-runtime-compose/build.gradle
M lifecycle/lifecycle-runtime-compose/src/androidTest/java/androidx/lifecycle/compose/DropUnlessLifecycleTest.kt
M lifecycle/lifecycle-runtime-compose/src/androidTest/java/androidx/lifecycle/compose/LifecycleEffectTest.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/DropUnlessLifecycle.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/FlowExt.kt
M lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/LifecycleEffect.kt
A lifecycle/lifecycle-runtime-compose/src/main/java/androidx/lifecycle/compose/LocalLifecycleOwner.kt
M wear/compose/integration-tests/navigation/build.gradle
Description
Background
The current Android architecture guidance recommends collecting flows in the UI in a lifecycle-aware manner. This allows the app to save expensive resources when not needed, for example, when the app is in the background.
Keeping expensive resources (e.g. Firabase queries, location or network updates, etc.) alive when not needed could impact the performance of the app and user's device.
To make it easier for developers to comply with the recommended best practices, APIs that do this should be added to the Lifecycle library.
Limitations with StateFlow
Ideally, there should exist APIs for both
Flow
andStateFlow
types. The latter is important since most flows consumed in Compose code will be of that type. If a helper function forStateFlow
doesn't exist, the UI (Compose code) would need to give those new functions an initial value (same as with Compose'sproduceState
or Flow'sstateIn
operator). That violates layering responsibilities since deciding the initial value belongs to the ViewModel/state holder, not the UI.However, as stated in the refdocs , the
StateFlow
interface is not stable for inheritance in 3rd party libraries, which makes it a tough call to create extensions functions on the type even though only the currentvalue
would be used.Proposed implementation #1
The
collectAsStateWithLifecycle
composable function is added to a newlifecycle-runtime-compose
artifact.Typically, this
collectAsStateWithLifecycle
API would be used like:Proposed implementation #2
To avoid passing the
initialValue
every time forStateFlow
types, another alternative could be therememberStateWithLifecycle
API. This API would also be added to a newlifecycle-runtime-compose
artifact.This would be used as follows:
This API would also offer a method overload that can take a
Flow
and aninitialValue
as parameters.