项目作者: PandoraMedia

项目描述 :
Android Bottom Navigation multiple stack manager
高级语言: Kotlin
项目地址: git://github.com/PandoraMedia/BottomNavigator.git
创建时间: 2019-08-02T21:15:45Z
项目社区:https://github.com/PandoraMedia/BottomNavigator

开源协议:Apache License 2.0

下载


Build Status

Bottom Navigator

Bottom Navigator is a library for managing multiple Fragment backstacks
controlled by the tabs of a BottomNavigationView. It has a simple API
and is feature-rich.

Unlike Material Design’s Android recommendations, the state of each
tab’s stack is preserved as the user switches between tabs.

The library keeps a history of previous tabs so that when the current
tab’s stack is exhausted the system back button will navigate to the
previously selected tab. Rotation is automatically handled and all
backstacks are preserved.

Table of Contents

How to Use it
Multiple taps on a tab
Detachability
Obtaining a BottomNavigator reference
Dependencies
Gradle

How to Use it

BottomNavigator needs an Activity with a BottomNavigationView and a
fragment container.

  1. <LinearLayout
  2. android:layout_width="match_parent"
  3. android:layout_height="match_parent"
  4. android:orientation="vertical">
  5. <FrameLayout
  6. android:id="@+id/fragment_container"
  7. android:layout_width="match_parent"
  8. android:layout_height="0dp"
  9. android:layout_weight="1" ></FrameLayout>
  10. <com.google.android.material.bottomnavigation.BottomNavigationView
  11. android:id="@+id/bottomnav_view"
  12. android:layout_width="match_parent"
  13. android:layout_height="56dp"
  14. app:menu="@menu/navigation_items" ></com.google.android.material.bottomnavigation.BottomNavigationView>
  15. </LinearLayout>

Then, in the activity’s onCreate you initialize the BottomNavigator by
calling its onCreate function. You need to provide a
rootFragmentsFactory that maps the BottomNavigationView’s menu items
to the root fragment for each tab. And a defaultTab which is the first
tab shown to the user.

  1. override fun onCreate(savedInstanceState: Bundle?) {
  2. super.onCreate(savedInstanceState)
  3. setContentView(R.layout.activity_main)
  4. navigator = BottomNavigator.onCreate(
  5. fragmentContainer = R.id.fragment_container,
  6. bottomNavigationView = findViewById(R.id.bottomnav_view),
  7. rootFragmentsFactory = mapOf(
  8. R.id.tab1 to { RootFragment1() },
  9. R.id.tab2 to { RootFragment2() },
  10. R.id.tab3 to { RootFragment3() }
  11. ),
  12. defaultTab = R.id.tab2,
  13. activity = this
  14. )
  15. }

Finally, wire up the back button to the BottomNavigator.

  1. override fun onBackPressed() {
  2. if (!navigator.pop()) {
  3. super.onBackPressed()
  4. }
  5. }

You can now call methods like navigator.addFragment(fragment) to add a
fragment to the current tab’s stack, or navigator.switchTab(R.id.tab1)
to switch stacks.

Checkout the sample app for a complete example.

Multiple taps on a tab

When a user taps on the currently shown bottom tab the expectation is
that the tab’s state will be reset. If the tab has a backstack then the
stack will be removed leaving the root fragment in whatever state it was
in. Another tap when the root Fragment is shown will recreate the fragment
and replace it with the new root fragment.

In order to get a smoother user experience you can avoid recreating the
root fragment by subscribing to BottomNavigator’s
resetRootFragmentCommand() rx stream. In the subscription to that stream
you should reset the given root fragment’s state. For example, you can
call smoothScrollToPosition(0) on a RecyclerView in the Fragment, or
you might clear the text in a search box.

Detachability

By default as Fragments are hidden and shown as the user navigates
around, the fragments are attached and detached which generates the
onDestroyView/onCreateView lifecycle on the Fragments. This is good
for memory consumption because it allows View objects that are not being
seen by the user to be garbage collected and then recreated in
onCreateView when the user wants to see them again.

If you do not want hidden Fragments to have their Views destroyed, maybe
because of lifecycle issues or for performance, you can mark a Fragment
as not being detachable. This will cause the fragments to be
hidden/shown without having their Views destroyed. To do this specify
detachable = false when adding the fragment, or for root Fragments by
initializing BottomNavigator with onCreateWithDetachability and
providing FragmentInfo objects to the rootFragmentsFactory.

Obtaining a BottomNavigator reference

BottomNavigator is scoped to an activity session, this means that after
rotation a new Activity gets the same instance from
BottomNavigator.onCreate as the previous Activity instance. Fragments
and other objects with reference to the Activity can obtain the same
activity session’s navigator instance by calling
BottomNavigator.provide(activity). This allows the Fragments to add
other Fragments to the BottomNavigator.

Dependencies

BottomNavigator works with AndroidX apps. It is built in Kotlin on top
of Architecture Components and RxJava.

Gradle

  1. dependencies {
  2. implementation 'com.pandora.bottomnavigator:bottom-navigator:1.8'
  3. }

Further reading

https://engineering.pandora.com/announcing-bottom-navigator-64f6e426a6b1

License

  1. Copyright 2019 Pandora Media, LLC
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. See accompanying LICENSE file or you may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.