Status Update
Comments
am...@google.com <am...@google.com>
am...@google.com <am...@google.com> #2
ca...@gmail.com <ca...@gmail.com> #3
ki...@gmail.com <ki...@gmail.com> #4
am...@google.com <am...@google.com> #5
[Deleted User] <[Deleted User]> #6
lo...@gmail.com <lo...@gmail.com> #7
t....@gmail.com <t....@gmail.com> #8
- Changing tab with the bottom bar resets the state and backstack of the previous tab/fragment, so our users are completely losing their navigation history (this is correctly implemented in Google Photos app)
- Pressing back from the root of one tab shows the initial tab/fragment instead of leaving the app"
[Deleted User] <[Deleted User]> #9
[Deleted User] <[Deleted User]> #11
ch...@google.com <ch...@google.com> #12
[Deleted User] <[Deleted User]> #13
ja...@gmail.com <ja...@gmail.com> #14
Feel free to check out the AndroidX source code and contribution instructions at
ch...@google.com <ch...@google.com> #15
"Bottom navigation behaves differently on Android and iOS. When you select a bottom navigation item (one that’s not currently selected), each platform displays different outcomes:
On Android: the app navigates to a destination’s top-level screen. Any prior user interactions and temporary screen states are reset, such as scroll position, tab selection, and in-line search.
On iOS: the destination reflects the user’s prior interaction. If the user previously visited that section of the app, they return to the last screen viewed (with its prior state preserved, if possible). Otherwise, the app navigates to the top-level screen.""
[Deleted User] <[Deleted User]> #16
> Default platform navigation can be overridden when needed to improve the user experience. For example, an Android app that requires frequent switching between sections can preserve each section’s state. Or, an iOS app can return users to the top-level screen (or reset their scroll position) if it better suits the use case.
I think providing an easy option to save each section's state is still worth providing as an option."
ch...@google.com <ch...@google.com> #17
am...@google.com <am...@google.com> #18
ar...@google.com <ar...@google.com> #19
az...@gmail.com <az...@gmail.com> #20
dg...@gmail.com <dg...@gmail.com> #21
[Deleted User] <[Deleted User]> #22
jo...@gmail.com <jo...@gmail.com> #23
dr...@gmail.com <dr...@gmail.com> #24
fa...@gmail.com <fa...@gmail.com> #25
One very conscious decision we made for Navigation 1.0.0 was to depend on the 28.0.0 Support Library. This means that, out of the box, we are limited to the APIs and behavior contained within. Fragments, the default NavHost+Navigator implementation provided out of the box, does not support multiple back stacks within a single FragmentManager. Even more unfortunately, Fragments do not offer an API for a system built on top of it (such as Navigation) to access the saved instance state and non configuration instance state (i.e., ViewModels) needed to write our own multiple back stack implementation while still using a single FragmentManager.
With that in mind, we'll be doing two things:
1. Short term: provide a public sample that shows our recommended implementation of multiple back stacks with Navigation using the current APIs (i.e., a separate NavHostFragment and navigation graph for each bottom navigation item). This is a top priority for us right now, only superseded by fixing issues blocking the 1.0 release.
2) Medium term: build the correct API on Fragments so that they can properly support multiple simultaneous back stacks, including properly saving and restoring saved instance state and non configuration instance state for all of the Fragments on all of the back stacks. This work is currently in an exploratory stage and, while I'm hopeful, I cannot offer a timeline or promise that this work will succeed.
Immediately following the 1.0 stable release, Navigation will move to depend on AndroidX (similar to how Paging and other Architecture Components switched to AndroidX), allowing us to be in a position to take advantage of the work in #2 to bring multiple back stack support to Navigation in a way that supports both Fragments as well as offers a generic API for other Navigators to provide the same functionality."
ps...@gmail.com <ps...@gmail.com> #26
dr...@gmail.com <dr...@gmail.com> #27
Regarding the short term plan: Have you published a public sample of how to implement this yet? If so can you provide a link?
dr...@gmail.com <dr...@gmail.com> #28
al...@google.com <al...@google.com> #29
This sample uses multiple NavHostFragments, one for each bottom navigation tab, to work around the current limitations of the Fragment API in supporting multiple back stacks.
We'll be proceeding with the Fragment API to support multiple back stacks and the Navigation API to plug into it once created, which will remove the need for anything like the NavigationExtensions.kt file. We'll continue to use this issue to track that work.
am...@google.com <am...@google.com> #30
There is an issue though: Clicking on Bottom nav tab should bring you back to the back to the first screen in the nav graph,
Right now clicking on bottom nav tab doesn't do anything, the only way back is to click back arrow on the toolbar
Thanks :)
ps...@gmail.com <ps...@gmail.com> #31
The Material Design guidelines do not specify any default behavior for reselected a tab:
de...@gmail.com <de...@gmail.com> #32
de...@gmail.com <de...@gmail.com> #33
pr...@gmail.com <pr...@gmail.com> #34
Kotlin is first class language, but even if you don't write in Kotlin, it's not difficult to convert the sample to Java (logic is the same)
ay...@gmail.com <ay...@gmail.com> #36
1) Nav Door Item 1
Bottom Nav Item1, Bottom Nav Item 2
2) Nav Door Item 2
Bottom Nav Item3, Bottom Nav Item 4
Navigating across top level multi stack retaining states for both top level and child level multi-stacks.
9g...@gmail.com <9g...@gmail.com> #37
Sorry if I am missing something trivial here but I took a look at the NavigationAdvancedSample and it does not mention
how one could handle navigation to items created by onCreateOptionsMenu. I tried it out but couldn't get it to work.
Could the workaround sample be updated to show how that might be achieved? or if someone has a clue :)
[Deleted User] <[Deleted User]> #38
kotlin.TypeCastException: null cannot be cast to non-null type androidx.navigation.fragment.NavHostFragment
at net.telekom.navigationtest.NavigationExtensionsKt$setupItemReselected$1.onNavigationItemReselected(NavigationExtensions.kt:186)
what are the requirements to implement this ?
za...@gmail.com <za...@gmail.com> #39
sh...@gmail.com <sh...@gmail.com> #40
ja...@gmail.com <ja...@gmail.com> #41
fixed it by making sure that each menu item @+id matches the file name of
the corresponding navigation graph
On Thu, Apr 4, 2019 at 6:54 AM <buganizer-system@google.com> wrote:
de...@gmail.com <de...@gmail.com> #42
#37 this is something that is done relatively easy if the fragment you want to navigate to, is included in the corresponding navigation_graph.xml
So under this assumption, someone should provide setHasOptionsMenu(true) in onCreateView and setup the desired menu.xml in onCreateOptionsMenu. Then in onOptionsItemSelected you can easily perform the following navigation by
val navController = findNavController()
navController.navigate(R.id.action_id)
However, be aware that at the moment, the navigation component does not set the menu items properly for each fragment. If a fragment has an options menu and then the others does not clear the menu at the time they get (re)created then the options menu wrongly continues to be visible.
In addition, I would like to know if there is a workaround on how someone could navigate to a fragment that has to be common for all the navigation graphs, in the advanced navigation sample. To be provide more context, lets assume a screen like global app "search" that the user is supposed to access by clicking on a menu item. At the current implementation of the sample someone would have to include the "search" fragment in every graph, define the appropriate action and implement my proposition to #38. Although this feels wrong and defeats the purpose of navigation component IMHO.
It would be great, if someone could provide me some feedback either in the aforementioned logic or possible workaround
pe...@gmail.com <pe...@gmail.com> #43
pe...@gmail.com <pe...@gmail.com> #44
hu...@gmail.com <hu...@gmail.com> #45
[Deleted User] <[Deleted User]> #46
an...@gmail.com <an...@gmail.com> #47
But there are some fragments (with complex layout structure) that the cost of rebuilding them is higher than their occupied Memory cost
hi...@gmail.com <hi...@gmail.com> #48
[Deleted User] <[Deleted User]> #49
1- The app crashes sometimes(I couldn't find why)
2- If you have common destinations in your graph you should add them in all graphs otherwise your app will crash. For example, if your app shows product destination in graph1, and if the user is on graph2 and he/she clicks on a notification that opens product destination(product destination is in graph1), your app will crash because graph2 does not have product destination
I just waste 1 week of my time, with this ugly component
ha...@gmail.com <ha...@gmail.com> #50
ri...@gmail.com <ri...@gmail.com> #51
fa...@gmail.com <fa...@gmail.com> #52
Component used: Navigation
Version used: 2.0.0, 2.1.0-alpha04
Devices/Android versions reproduced on: Any
If this is a bug in the library, we would appreciate if you could attach:
- Sample project to trigger the issue (please see attached project)
How to reproduce:
In the current project
1. Create a deeplink for navigation to About fragment ( adb shell am start -W -a android.intent.action.VIEW -d "
2. Click on one of the buttons (REGISTER or BLANK) in About fragment in order to navigate to corresponding destination.
3. App crashes with an exception: java.lang.IllegalArgumentException: navigation destination com.example.android.navigationadvancedsample:id/action_blankFragment is unknown to this NavController.
de...@gmail.com <de...@gmail.com> #53
sh...@google.com <sh...@google.com> #55
za...@gmail.com <za...@gmail.com> #56
dr...@gmail.com <dr...@gmail.com> #57
sh...@google.com <sh...@google.com> #58
sp...@gmail.com <sp...@gmail.com> #59
sh...@google.com <sh...@google.com> #60
ha...@gmail.com <ha...@gmail.com> #61
je...@gmail.com <je...@gmail.com> #62
pr...@gmail.com <pr...@gmail.com> #63
missing in current navigation component.
But anyway I'd like to see someone is working on it.
Regards,
Luke Shan
Email: shanyuliang@gmail.com
* Started Android development since 2009;
* 6 years' experience on Mainframe platform;
* 2 years' experience on Microsoft platform;
* Certified DB2 V8.1 Administrator for z/OS, Windows and Linux and
Certified Java Programmer for Java SE5;
On Tue, Sep 24, 2019 at 9:41 AM <buganizer-system@google.com> wrote:
su...@gmail.com <su...@gmail.com> #64
sh...@google.com <sh...@google.com> #65
As per the public hotlists, this is expected as part of Fragment 1.3 and Navigation 2.3. That's what we're continuing to work towards.
ny...@gmail.com <ny...@gmail.com> #66
fa...@gmail.com <fa...@gmail.com> #67
Please provide a general solution not just the BottomNavigationBar
le...@gmail.com <le...@gmail.com> #68
- A Fragment specific API to ensure that the state of Fragments on multiple back stacks is saved
- A NavController API that provides the generic framework that allows any Navigator (including FragmentNavigator) to support multiple back stacks
- New APIs in NavigationUI to allow you to control whether setupWithNavController() uses a single back stack style model (the current model) or multiple back stacks when using a BottomNavigationView or NavigationView
So if you have custom UI elements or want to have more direct control, you'll be able to use the NavController level APIs.
pi...@gmail.com <pi...@gmail.com> #69
yi...@gmail.com <yi...@gmail.com> #70
If the answer is yes, an example of how to handle the following use case would be welcome:
BottomNavView with fragments A and B. Both can navigate to a fragment C. From there on out the navigation would look exactly the same. Will it be possible to handle this case with a single nav.xml but multiple back stacks?
de...@gmail.com <de...@gmail.com> #71
We'd still recommend using <include> tags and splitting up your graph into maintainable portions though as per the documentation:
t....@gmail.com <t....@gmail.com> #72
cn...@gmail.com <cn...@gmail.com> #73
za...@gmail.com <za...@gmail.com> #74
FATAL EXCEPTION: main
Process: com.xxx.xx.develop, PID: 20784
java.lang.IllegalStateException: Fragment HomeFragment{ca12844} (f2a95b7f-5df3-4106-b55b-b09067a8accb)} not associated with a fragment manager.
at androidx.fragment.app.Fragment.getParentFragmentManager(Fragment.java:945)
at androidx.navigation.fragment.NavHostFragment.findNavController(NavHostFragment.java:107)
at androidx.navigation.fragment.FragmentKt.findNavController(Fragment.kt:29)
at com.principles.pia.ui.home.HomeFragment$setUpViewModel$1.invoke(HomeFragment.kt:83)
at com.principles.pia.ui.home.HomeFragment$setUpViewModel$1.invoke(HomeFragment.kt:32)
at com.principles.pia.ui.home.items.BookItem$bind$$inlined$apply$lambda$1.onClick(BookItem.kt:22)
at android.view.View.performClick(View.java:6256)
at android.view.View$PerformClick.run(View.java:24701)
at android.os.Handler.handleCallback(Handler.java:789)
at android.os.Handler.dispatchMessage(Handler.java:98)
at android.os.Looper.loop(Looper.java:164)
at android.app.ActivityThread.main(ActivityThread.java:6541)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)
Does this look like a problem that is being tracked by this
For example we have two bottom nav tabs, per AdvancedSample, two nav graphs, each containing the following destinations: [A1, A2, A3] and [B1, B2, B3] (where A1 and B1 are corresponding starDestinations). If the app needs to be able to navigate A2->B3, would it be possible?
da...@gmail.com <da...@gmail.com> #75
[Deleted User] <[Deleted User]> #76
ro...@gmail.com <ro...@gmail.com> #77
Welcome to 2020, multiple back stacks issue. Throughout this update I'll be using Markdown syntax, so make sure to hit the 'Use the new UI' button on the top of the page to switch over to the new UI and the markdown goodness.
As mentioned in 2.3.0-alpha01
and the required support in Fragment 1.3.0-alpha01
once the FragmentManager
), fixing that has exposed a number of other issues so far (many of which were released in
However, this additional layer of work was not anticipated in the above 2.3.0-alpha01
Instead, we'll be releasing Navigation 2.3.0-alpha01
without multiple back stack support. Our plan is to finish up the Fragment rework in the upcoming Fragment 1.3.0-alpha01
release and ensure that everything works better than ever in the single back stack world that is FragmentManager
today. Oh, and get Navigation 2.3.0 to stable as soon as possible.
I've updated the public hotlists to denote that multiple back stack support is now tentatively scheduled for Navigation 2.4.0-alpha01
and Fragment 1.4.0-alpha01
, respectively. If everything goes better than expected, it may be earlier than that.
Please be considerate when replying to this issue. It does notify everyone who has starred this issue.
an...@googlemail.com <an...@googlemail.com> #78
sh...@google.com <sh...@google.com> #79
fr...@gmail.com <fr...@gmail.com> #80
sh...@google.com <sh...@google.com> #81
zz...@gmail.com <zz...@gmail.com> #82
I tried the sample but its not cutting it.. Having this baked into the navigation graphs would make life so much better!
sh...@gmail.com <sh...@gmail.com> #83
sh...@gmail.com <sh...@gmail.com> #84
ss...@gmail.com <ss...@gmail.com> #85
lo...@gmail.com <lo...@gmail.com> #86
Please be considerate when replying to this issue, because hundreds of people will get emails for unhelpful contributions.
am...@gmail.com <am...@gmail.com> #87
am...@google.com <am...@google.com>
pa...@gmail.com <pa...@gmail.com> #88
So please for the love of God read and don't reply for asking status update or any irrelevant things. You're notifying hundreds of people that starred this issue for meaningless info.
we...@gmail.com <we...@gmail.com> #89
My solution:
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val appBarConfigurationIds = setOf(R.id.fragmentTab1, R.id.fragmentTab2, R.id.fragmentTab3, R.id.fragmentTab4)
val appBarConfiguration = AppBarConfiguration(appBarConfigurationIds)
setupActionBarWithNavController(navController, appBarConfiguration)
bottomNavigationView.setOnNavigationItemSelectedListener { menuItem ->
if (menuItem.itemId == navController.currentBackStackEntry?.destination?.id)
return@setOnNavigationItemSelectedListener false
appBarConfigurationIds.forEach { fragmentId ->
if (fragmentId == menuItem.itemId) {
navController.navigate(fragmentId, null, NavOptions.Builder() <--- M
.setLaunchSingleTop(true) <--- A
.setPopUpTo(navController.currentDestination!!.id, true) <--- G
.build()) <--- I
return@setOnNavigationItemSelectedListener true <--- C
}
}
false
}
navController.addOnDestinationChangedListener { controller, destination, arguments ->
if (appBarConfigurationIds.indexOf(destination.id) != -1) {
controller.graph.startDestination = destination.id// <--- MAGIC
}
}
}
override fun onSupportNavigateUp() = navController.navigateUp()
private var lastBackPressed = 0L
override fun onBackPressed() {
if (navHostFragment.childFragmentManager.backStackEntryCount == 0) {
if (lastBackPressed + 1000 > System.currentTimeMillis()) {
super.onBackPressed()
} else {
lastBackPressed = System.currentTimeMillis()
toast("Double tap for exit")
}
} else
super.onBackPressed()
}
}
kp...@gmail.com <kp...@gmail.com> #90
sh...@google.com <sh...@google.com> #91
sh...@google.com <sh...@google.com> #92
pa...@outlook.com <pa...@outlook.com> #93
There's a workaround provided the
I'd like to know any updates from Google to officially support this kinda very common requirement, thanks
sh...@google.com <sh...@google.com> #94
As per
As seen on the 1.3.0-rc02
release before the stable release, thus clearing the way entirely for multiple back stacks (as mentioned in
Our own internal API design process has reached the point where we have a firm direction and you should begin to see changes come through attached to this bug as we build up to the final APIs to support multiple back stacks at the Fragment and Navigation level.
This will include:
- The Fragment APIs for supporting multiple back stacks (thus enabling multiple back stacks even if you do not use Navigation)
- The generic Navigation API that will allow any Navigator (be it Fragments, Compose, etc) to support multiple back stacks
- Rework of the existing Navigators to support multiple back stacks out of the box
- The APIs in
NavigationUI
and other surfaces to allow you to easily opt in to using multiple back stacks
I'd ask that you hold your comments on the intermediate steps along the way as much of it will make more sense when taken as a whole once we release the first alphas (still planned for Fragment 1.4.0-alpha01
and Navigation 2.4.0-alpha01
). Definitely expect a more thorough run through of how to use of these APIs via documentation, samples (including updating the NavigationAdvancedSample
to use the new APIs), blog posts, videos, and everything else at that point and onward. Besides seeing each commit come through on this issue, we'll also make sure to point out when a first snapshot is available with working APIs for those of you who would like an early preview before the first alpha releases.
Again, please keep in mind the many people who get emailed for each comment on this issue - let's continue to keep this as a good channel for updates and not for general discussion or requests for additional progress updates.
pa...@outlook.com <pa...@outlook.com> #95
Branch: androidx-main
commit ce3bc9455ff4eb607601b5a417f5248e93507381
Author: Ian Lake <ilake@google.com>
Date: Thu Jan 21 15:47:27 2021
Track whether an Op is associated with the topmost fragment
Multiple back stacks involve saving and later restoring the
set set of FragmentTransactions that make up the back stack
(via addToBackStack()). The replace() operation is a
unique case where you don't need to know what fragment you
are replacing - it just operates on the topmost fragment
for that container.
Due to how expandOps() works, that initial OP_REPLACE
is lost, replaced by the more primitive operations
(namely, OP_UNSET_PRIMARY_NAV, OP_REMOVE, and OP_ADD).
However, in order to save that FragmentTransaction for
later restoration in the multiple back stack case, we
need to know that the OP_REMOVE doesn't apply to a
specific fragment, but instead just the topmost
fragment of the container.
By tracking this information, we ensure that
you can safely restore a set of FragmentTransactions
(i.e., a back stack) onto the FragmentManager even
if changes have been made between saving and restoring
that back stack.
As an example, let's say the you have an initial fragment
in a container of A and two FragmentTransactions that
use addToBackStack() that called replace() with B and C,
respectively. In the multiple back stack world, you could
save both of those FragmentTransactions and return to only
A. You could then create new FragmentTransactions with
addToBackStack() to replace A with D and E, and then later
restore the back stack previously saved. By doing so,
the FragmentTransaction that previously replaced A with B
would now replace E (the topmost fragment of the container)
with B, followed by a replacement of B with C. Thus, you
would be on a stack that had A -> D -> E -> B -> C.
There's still more work needed to build up the rest of the
Fragment APIs for multiple back stacks - this is just
the first step on the underlying implementation.
Test: existing tests pass
BUG: 80029773
Change-Id: I8432e9043eeb14c05b1a121d94e00722cff31688
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecord.java
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackState.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentTransaction.java
me...@google.com <me...@google.com> #96
Branch: androidx-main
commit 10604f16c11ed787d0f0e9ac94dc509532987d14
Author: Ian Lake <ilake@google.com>
Date: Mon Jan 25 17:44:05 2021
Add API for saving the back stack
The Fragment back stack is made up of separate
BackStackRecords, also known as transactions (as they
implement FragmentTransaction and that's the public
API used to create them). While a popBackStack()
operation will reverse the transaction and destroy
all of the no longer referenced fragments, to *save*
the state we need a new type of operation.
This operation is called saveBackStack() and takes the
name of the back stack entry (the one set via
addToBackStack()). This required converting some
common code (finding the correct index on the back stack)
so that it is useable by both pop and save.
This does not yet actually do any saving, nor is the
restore half of the equation built yet, but this is
the first step down that path.
Test: new SaveRestoreBackStackTest suite
BUG: 80029773
Relnote: N/A
Change-Id: I661f33c0ea7bb67338f7d9e75f8e8b8454d2bc21
M fragment/fragment/api/current.txt
M fragment/fragment/api/public_plus_experimental_current.txt
M fragment/fragment/api/restricted_current.txt
A fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
yq...@gmail.com <yq...@gmail.com> #97
Branch: androidx-main
commit b95c6d3b57d048a965c9062f9ae595a88a034e3d
Author: Ian Lake <ilake@google.com>
Date: Mon Feb 01 15:58:33 2021
Ensure saveBackStack() is self contained
A key part to saving a back stack is to be
able to later restore that same back stack.
As such, we need to ensure that the
FragmentTransactions that are being saved
are self contained - i.e., that the only
fragments they specifically mention are
added as part of those transactions. By
ensuring that the saved transactions are
self contained, we can ensure that they
can be correctly restored even if the
underlying state of the FragmentManager
is changed between the save and restore
operations.
This builds upon the previous topmost
fragment change so as to continue to allow
operations like replace() which don't
explicitly mention previously added fragments.
As part of this, generateOpsForPendingActions() has
been made more resilient to crashes - if a
generateOps() call crashes, it still throws the
exception, but the set of pending actions that
crashed FragmentManager will now be cleared out
so that they are not reprocessed any time later
if something (like our test suite...) catches
exceptions thrown by executePendingOperations().
Test: new savePreviouslyReferencedFragment test
BUG: 80029773
Change-Id: I0bde3c27f415990c2918d98a79ea76c3d0c6b55e
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
pv...@gmail.com <pv...@gmail.com> #98
I have been using the workaround proposed for using multiple back stacks navigation in my app - [CheSScan](
Thanks!
vz...@gmail.com <vz...@gmail.com> #99
Branch: androidx-main
commit d0752ab5b18e85ac5d45f768bce8ad0a8ed36c3b
Author: Ian Lake <ilake@google.com>
Date: Thu Feb 11 14:39:52 2021
Ensure all saved transactions have reordering allowed
The documentation specifically calls out that
setReorderingAllowed(true) should be used on each
FragmentTransaction (see
but this is moves from a 'should use' to a
'must use' pattern when it comes to saving the
back stack and the FragmentTransactions on it.
This is because setReorderingAllowed(true) is
critical to ensuring that a set of back to back
transactions are executed as an atomic operation,
thus preventing any intermediate fragments (those
that are added and then moved onto the back stack)
from moving through lifecycle states unnecessarily.
By checking this when executing saveBackStack(), we
ensure that the later restore operates as expected.
Test: new saveNonReorderingAllowedTransaction test
BUG: 80029773
Change-Id: Ied3b2a221202ca4626975853b741c76edd08f978
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
sh...@google.com <sh...@google.com> #100
Branch: androidx-main
commit fe00de00ab8a008405810a7f1ef3c9826d5c3cd7
Author: Ian Lake <ilake@google.com>
Date: Thu Feb 18 16:57:47 2021
Add BackStackState infrastructure
A saved back involves two main components:
1. The set of FragmentTransactions that make
up that saved back stack
2. The set of Fragments that were added
as part of that back stack
This is encapsulated in the new BackStackState
class. This is class that will eventually be
created by saveBackStack() and later used by
restoreBackStack() to restore the stack. Those
integration points are coming in follow up CLs:
this is solely the Parcelable used to store this
state.
Test: new BackStackRecordTest test suite
BUG: 80029773
Change-Id: I1658c01fdf7af8941436db96000e91796b823108
A fragment/fragment/src/androidTest/java/androidx/fragment/app/BackStackRecordTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecordState.java
A fragment/fragment/src/main/java/androidx/fragment/app/BackStackState.java
sh...@google.com <sh...@google.com> #101
Branch: androidx-main
commit 7db1719dba9c8646c9dbcda7bf0a61926c1789df
Author: Ian Lake <ilake@google.com>
Date: Thu Feb 25 15:42:37 2021
Disallow retained fragments in saveBackStack()
When saving the back stack, all fragments have
their state and non-config state saved before they
are fully destroyed. This, by necessity, also means
fully destroying any child fragments (and their
child fragments, etc.), saving the state of those as
well. This state is then held until restoreBackStack()
is called (or the entire FragmentManager is
permanently destroyed).
This state-only saving means that retained fragments,
either directly or as child fragments, are not supported
as their contract is that they are not destroyed
until they are permanently removed. Instead, developers
should use ViewModels to store any non-config state.
Test: new SaveRestoreBackStackTest tests
BUG: 80029773
Change-Id: I01f94cbfae3f3850e8a4efe84e6576e408f68ef8
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
sa...@gmail.com <sa...@gmail.com> #102
Branch: androidx-main
commit f54b6d3d78139605391592d689f2f3c5c9e55197
Author: Ian Lake <ilake@google.com>
Date: Thu Feb 25 17:19:43 2021
Add infra for saving/restoring BackStackStates
Each FragmentManager is responsible for tracking
the set of BackStackState objects that have been
saved via saveBackStack() and the associated name.
This state needs to be saved and restored alongside
the rest of the FragmentManager's state to ensure
that the state is available for later restoration
even after one or more configuration changes.
This adds the underlying infrastructure for saving
and restoring the state objects that will later be
populated by saveBackStack().
Test: existing tests that save/restore state still pass
BUG: 80029773
Change-Id: I5776c50143deb4d7679c581a2e7655d5c2655443
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManagerState.java
[Deleted User] <[Deleted User]> #103
[Deleted User] <[Deleted User]> #104
sh...@google.com <sh...@google.com> #105
te...@gmail.com <te...@gmail.com> #106
Branch: androidx-main
commit a7835d060734f7a3dc7acc457c822330ef64d5a2
Author: Ian Lake <ilake@google.com>
Date: Mon Mar 15 16:54:08 2021
Build infrastructure for collapsing transactions
Right before a FragmentTransaction is executed,
operations like replace() and setPrimaryNavigationFragment()
are expanded into multiple operations:
- A replace() becomes one or more remove() operations
plus an add() (unless you specifically call add()
earlier in the transaction...for some reason).
- setPrimaryNavigationFragment() becomes an unset of the
previous primary navigation fragment and then the set
of the new primary navigation fragment
These expanded operations can reference previously
created fragments, despite them not being explicitly
mentioned when building the FragmentTransaction. When
it comes to saving and later restoring these transactions,
we need to collapse the transactions back down into their
original form so that:
1) they don't reference other fragments not being saved
2) they can be re-expanded later onto a new back stack
This CL swaps out the term 'topmost fragment' with
'from expanded op' to better mark which operations
came from one of these expansions. This then allows
us to write the infrastructure and tests to ensure
that a collapse puts the transaction back into
the previous state that will later be used by
saveBackStack().
Test: updated and new BackStackRecordTest tests pass
BUG: 80029773
Change-Id: Iadd95c557b759bd95f07d89cadf62b44d4971956
M fragment/fragment/src/androidTest/java/androidx/fragment/app/BackStackRecordTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecord.java
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecordState.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentTransaction.java
sh...@google.com <sh...@google.com> #107
I am facing issue with explicit deeplinking, which is not launching the respected screen, Any update on this issue, or any workaround.
ma...@gmail.com <ma...@gmail.com> #108
ma...@gmail.com <ma...@gmail.com> #109
Branch: androidx-main
commit cac5a59e08bff1fc47a721d112fa5289d3ddf5bc
Author: Ian Lake <ilake@google.com>
Date: Thu Mar 25 17:43:49 2021
Make saveBackStack() actually save the back stack state
The saveBackStack() API is responsible for a number of
things:
1) Popping the saved FragmentTransactions off the back
stack - this was done in previous CLs.
2) Saving the actual FragmentTransactions themselves
so that they can later to restored
3) Save the state of each Fragment included in
those FragmentTransactions
This change seeks to address the core points of #2
and #3. Now, when saveBackStack() is called, all
Fragments will go through onSaveInstanceState(), either
immediately after onStop() (mirroring the behavior of
the activity being torn down due to a configuration
change, etc.) or immediately before onDestroy() (in
cases where the Fragment is already stopped - such as
when it is on the back stack).
The FragmentTransactions themselves are also
collapsed (i.e., replacing the expanded set of
REMOVE+ADD operations with the original REPLACE operation)
and have their state saved, readying them for later
restoration.
Note that this change does *not* handle nonconfig state
(ViewModels). That will be done in a follow up CL.
Test: saveBackStack test passes
BUG: 80029773
Change-Id: I37327ab6a7deab0d72d5be1a251c0d5e3a280362
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecord.java
M fragment/fragment/src/main/java/androidx/fragment/app/Fragment.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentStateManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentTransaction.java
sh...@google.com <sh...@google.com> #110
Branch: androidx-main
commit 5104c7c386ea69551616e8723cb87e5dbd27f6ad
Author: Ian Lake <ilake@google.com>
Date: Mon Mar 29 13:43:30 2021
Add the API to restore a saved back stack
Saving the back stack is only half of the API;
the other half is the ability to later restore
that back stack onto the FragmentManager. This
is the role of the restoreBackStack() API. It
takes a name (the same name passed to
addToBackStack() and then later passed to
saveBackStack()) and re-instantiates all of the
FragmentTransaction and Fragment instances and
re-applies them to the FragmentManager.
Particularly, this builds on the state saving
done by saveBackStack() to ensure that every
fragment that was saved has its saved state
restored when you call restoreBackStack().
The addition of this API did expose some issues
in the previously added code, namely:
- saveBackStack() needs a callback immediately
after the FragmentTransaction pop is executed and
*before* the Fragments are destroyed. This ensures
that the right internal IDs of Fragments are saved
in the BackStackRecordState and means we can't use
the on commit runnables list and need to maintain
our own list.
- As the population of the BackStackRecordState
is asynchronous, BackStackState needs to keep a
reference to the original list created by
saveBackStack() rather than copying the entries
into its own ArrayList.
Relnote: N/A
Test: new tests pass
BUG: 80029773
Change-Id: Ib8c35cea758379374cde7c1253922e345544b3c5
M fragment/fragment/api/current.txt
M fragment/fragment/api/public_plus_experimental_current.txt
M fragment/fragment/api/restricted_current.txt
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackRecord.java
M fragment/fragment/src/main/java/androidx/fragment/app/BackStackState.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentTransaction.java
bo...@gmail.com <bo...@gmail.com> #111
Branch: androidx-main
commit 3645a0d738376c0035c2436ba5c53c750fe84d86
Author: Ian Lake <ilake@google.com>
Date: Tue Mar 30 16:02:51 2021
Save ViewModels when saving and restore the back stack
The most important part of calling saveBackStack() is
that the state of the fragments must be saved.
This can be roughly summarized in three parts:
1) The Fragment's saved instance state
2) The Fragment's View state
3) The Fragment's non-config state
The first two are already handled by the previous
changes, but the later one is just as important,
particularly when it comes to immediately showing
cached data. As retained fragments are not
supported when using saveBackStack(), the only
non-config state is that of ViewModels.
Besides just avoiding clearing the ViewModels when
the fragment is destroyed as part of saving the
back stack, special attention was taken to the
case where the Activity / parent Fragment is itself
destroyed. In this case, any ViewModels associated
with saved fragments must be specifically cleared as
they would not automatically be cleaned up on their
own (as their containing fragment no longer exists,
just its saved state).
Test: updated tests pass
BUG: 80029773
Change-Id: Idafc0af58e1a43893774f6ae6d7caac8bd56efb4
M fragment/fragment/src/androidTest/java/androidx/fragment/app/SaveRestoreBackStackTest.kt
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManager.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentManagerViewModel.java
M fragment/fragment/src/main/java/androidx/fragment/app/FragmentStateManager.java
Description
Version used: 26.0.2
Theme used: Theme.AppCompat.NoActionBar
Devices/Android versions reproduced on: AVD API 25
I just upgraded to API 26 and support library 26.0.2. But I found that my RecyclerView items is not clickable right after the scrolling. If you wait for a second, it will work. But if you click the item immediately, it won't. Even if the RecyclerView is not scrolling at all(e.g. has scrolled to the top).
When I downgraded to support library 25.4.0 everything goes fine again. The key point is that my RecyclerView is in a CoordinatorLayout and has a SCROLL_FLAG_SCROLL flag on my Toolbar of the AppBarLayout. If I don't use this flag, then this problem will disappear.
I've tried to add focusable="false" to the CoordinatorLayout but still had no luck.
Is there any way to disable this behavior? Because it's really annoying to click twice to trigger the click event.
I think the problem is the scrollState of the RecyclerView. When it's stopped scrolling, it's not changed to SCROLL_STATE_IDLE immediately. Looking into the source code of RecyclerView, I found there's a ViewFlinger controlling the scroll state. When I fling down to scroll to the top, it's not calling setScrollState(SCROLL_STATE_IDLE) immediately. Instead, it wait for a while to trigger this method. The more fast I fling, the more time I need to wait. It just like the RecyclerView is still scrolling in the background. Because the scroller.isFinished() doesn't return true right after the RecyclerView stop scrolling when it touched the top. Maybe it's a bug of the RecyclerView when it's in a CoordinatorLayout.
The attachment is a screen recording of this behavior.
<android.support.design.widget.CoordinatorLayout
android:id="@+id/coordinateLayout"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.design.widget.AppBarLayout
android:id="@+id/fragmentAppBar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:elevation="0dp"
android:background="@null">
<include
android:id="@+id/dynamicActionBarHolder"
layout="@layout/dynamic_action_bar"/>
</android.support.design.widget.AppBarLayout>
<android.support.v4.widget.SwipeRefreshLayout
android:id="@+id/pullToRefreshMailRecycler"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_behavior="@string/appbar_scrolling_view_behavior">
<android.support.v7.widget.RecyclerView
android:id="@+id/mailRecyclerView"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</android.support.v4.widget.SwipeRefreshLayout>
</android.support.design.widget.CoordinatorLayout>
layout/dynamic_action_bar.xml
<FrameLayout xmlns:android="
xmlns:app="
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
app:layout_scrollFlags="scroll"
android:clickable="true"
android:background="?theme.dynamicActionBarBackground">
<ImageButton
android:id="@+id/dynamicAcbMenuIcon"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:background="@drawable/article_explicit_button_background"
android:src="?theme.menuIcon"/>
<RelativeLayout
android:id="@+id/dynamicAcbTitleHolder"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="?attr/actionBarSize"
android:paddingTop="5dp"
android:paddingBottom="5dp"
android:paddingEnd="5dp"
android:layout_gravity="center_vertical">
<TextView
android:id="@+id/dynamicAcbTitle"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:singleLine="true"
android:textSize="@dimen/action_bar_title_portrait_size"
android:textColor="?theme.listItemTitleColor"
android:ellipsize="end"
android:text="ActionBar"/>
<TextView
android:id="@+id/dynamicAcbSubtitle"
android:layout_below="@+id/dynamicAcbTitle"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:singleLine="true"
android:textSize="@dimen/action_bar_subtitle_portrait_size"
android:textColor="?theme.listItemTitleColor"
android:ellipsize="end"
android:text="If you say so"/>
</RelativeLayout>
</FrameLayout>