Status Update
Comments
il...@google.com <il...@google.com> #2
The only currently supported strategy for adding Fragments to compose is via
il...@google.com <il...@google.com> #3
Yes, I am familiar with AndroidViewBinding, however, as far as I know it has some limitations, like, I can't decide at runtime what fragment to commit.
il...@google.com <il...@google.com> #4
I also can't give parameters to the fragment being inflated.
il...@google.com <il...@google.com> #5
Branch: androidx-main
commit ffe9335a155bf0dea9c3558e50da01d526579040
Author: Jeremy Woods <jbwoods@google.com>
Date: Tue Mar 26 20:24:54 2024
Create AndroidFragment to allow adding fragments in Compose
Adding new AndroidFragment API that creates the given Fragment class and
adds it to the enclosing FragmentManager, within Compose.
It handles setting the ViewCompositionStrategy and saving the state of
the fragment across recompositions.
RelNote: "The new AndroidFragment Composable allows adding fragments
into the Compose hierarchy via the fragment class name."
Test: added AndroidFragmentTest
Bug: 230126482
Bug: 312895363
Change-Id: Icf84199bbe487b2a2b6a95d2b6e09415f810e77a
M docs-tip-of-tree/build.gradle
M fragment/fragment-compose/api/current.txt
M fragment/fragment-compose/api/restricted_current.txt
M fragment/fragment-compose/build.gradle
A fragment/fragment-compose/samples/build.gradle
A fragment/fragment-compose/samples/src/main/java/androidx/fragment/compose/samples/FragmentComposeSamples.kt
M fragment/fragment-compose/src/androidTest/AndroidManifest.xml
A fragment/fragment-compose/src/androidTest/java/androidx/fragment/compose/AndroidFragmentTest.kt
A fragment/fragment-compose/src/androidTest/java/androidx/fragment/compose/FragmentRecreateTest.kt
A fragment/fragment-compose/src/androidTest/java/androidx/fragment/compose/FragmentRemoveTest.kt
A fragment/fragment-compose/src/androidTest/java/androidx/fragment/compose/SimpleEditTextFragment.kt
A fragment/fragment-compose/src/androidTest/java/androidx/fragment/compose/test/EmptyTestActivity.kt
A fragment/fragment-compose/src/androidTest/res/layout/content.xml
A fragment/fragment-compose/src/androidTest/res/layout/sample_edit_test_layout.xml
A fragment/fragment-compose/src/main/java/androidx/fragment/compose/AndroidFragment.kt
A fragment/fragment-compose/src/main/java/androidx/fragment/compose/FragmentState.kt
M fragment/fragment/api/current.txt
M fragment/fragment/api/restricted_current.txt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentTransaction.java
M settings.gradle
il...@google.com <il...@google.com>
ap...@google.com <ap...@google.com> #6
Hey, Thanks to checking this out, I just checked this sample in the the repository, but this does not solve my issue.
I don't have access to the actual fragment class, we use an abstraction that just returns me a instance of Fragment with the arguments set, so this won't solve the issue for me, would it be possible to allow passing the Fragment instance ?
ap...@google.com <ap...@google.com> #7
Re
This aligns with other fragment based APIs like FragmentScenario
and NavHostFragment
. You'll want to adjust your own abstraction to use those APIs and separate the creation of the fragment instance from the arguments.
ap...@google.com <ap...@google.com> #8
I'll note that FragmentFactory does allow you to request an AbstractBaseFragment
and redirect it to a specific subclass of that class without exposing that subclass itself if you do need that level of indirection.
il...@google.com <il...@google.com> #9
We were always allowed to create a new Fragment and to commit a Fragment instance in FragmentManager, could you align with that as well? My abstraction returns a Fragment instance because this allows us to avoid directly depending on the gradle module that provides the Fragment and this strategy improves compilation time by making the gradle dependency graph flatter.
sn...@gmail.com <sn...@gmail.com> #10
You'll want your abstraction to be implemented in terms of a FragmentFactory.
il...@google.com <il...@google.com> #11
My abstraction does not use fragmentFactory, and don't need to. How is that related to this issue? Devs have always been able to instantiate a fragment using default fragment constructor, I am not exposing which Fragment subclass is being instantiated in the calling site or using the className as key to find which fragment to instantiate, to add a fragment to FragmentManager we just need the base class Fragment.
ro...@theorytank.com <ro...@theorytank.com> #12
il...@google.com <il...@google.com> #13
Re AndroidFragment
with key(url)
to reset that subtree whenever url
changes.
[Deleted User] <[Deleted User]> #14
Thanks, using key(url)
works like a charm. I thought this function was only available for LazyList|Column
...
Description
Jetpack Compose release version: Compose alpha11 Android Studio Build: Arctic Fox Canary 4
There appears to be a number of issues with inflating a fragment by using
AndroidViewBinding
:If you use
FragmentContainerView
, the fragment appears the first time the container is inflated, but later times it does not appear.If you use
<fragment>
, the fragment appears the first time the fragment is inflated, but will crash if you try to re-inflate the same layout a second time. Rotating your screen or otherwise recreating the Activity will reset it so that the first inflation worksIn both cases, removing the
AndroidViewBinding
from your compose hierarchy does not remove the underlyingFragment
from theFragmentManager
. This means that the Fragment remains in memory forever afterwards.I've attached a sample project that contains three situation (using
FragmentContainerView
, using<fragment>
, and manually doing aFragmentTransaction
in Compose code) and a 'Show' button of each that lets you confirm what happens when theAndroidViewBinding
is removed from the hierarchy.My investigation lead to a couple of discoveries:
After a configuration change, the whole activity and the
FragmentManager
has moved to theRESUMED
state before the compose hierarchy is processed andAndroidViewBinding
is able to inflate its container. This means the original fragment added via the inflatedFragmentContainerView
has already had its view created and moved to theRESUMED
state before the container exists. Thus, the transition betweenonCreateView()
andonViewCreated()
where the view is attached to its container has already passed when theAndroidViewBinding
creates the container.The
<fragment>
tag added fragment never goes above theCREATED
state on its own - Fragments is correctly stopping it from going higher thanCREATED
during the inflation, but because Compose is inflating it after the FragmentManager has already moved throughRESUMED
, the usual moving of state for<fragment>
inflated fragments doesn't happen. Doing anotherFragmentTransaction
on a different fragment/container moves it toSTARTED
, but it never moves toRESUMED
as no FragmentTransactions on that container are pending (and moving toRESUMED
is done at a container by container basis) as the<fragment>
tag doesn't actually do aFragmentTransaction
.It is expected that when a composable is removed from the hierarchy that any internal state is thrown out (that's why you'd normally hoist state you want to save). Because Fragments are never removed when the
AndroidViewBinding
is removed from the hierarchy, the Fragments keep their state over a removal / addition of a new element that happens to inflate that same layout.