【问题标题】:BottomNavigationView back button behavior should work like the Youtube App but crashesBottomNavigationView 后退按钮行为应该像 Youtube 应用程序一样工作,但会崩溃
【发布时间】:2017-11-22 13:28:44
【问题描述】:

重现步骤:

  1. 启动一个新的Android项目,选择“BottomNavigationView”:
  2. 将 MainActivity 替换为:

    class MainActivity : AppCompatActivity() {
    
    private var fragmentIds = ArrayList<Int>()
    
    val fragmentA: FragmentA = FragmentA()
    private val fragmentB = FragmentB()
    private val fragmentC = FragmentC()
    
    private fun getFragment(fragmentId: Int): Fragment {
        when (fragmentId) {
            R.id.navigation_home -> {
                return fragmentA
            }
            R.id.navigation_dashboard -> {
                return fragmentB
            }
            R.id.navigation_notifications -> {
                return fragmentC
            }
        }
        return fragmentA
    }
    
    private fun updateView(fragmentId: Int) {
        var exists = false
        fragmentIds
                .filter { it == fragmentId }
                .forEach { exists = true }
    
        if (exists) {
            fragmentIds.remove(fragmentId)
            showTabWithoutAddingToBackStack(getFragment(fragmentId))
        } else {
            fragmentIds.add(fragmentId)
            showTab(getFragment(fragmentId))
        }
    }
    
    private val onNavigationItemClicked = BottomNavigationView.OnNavigationItemSelectedListener { item ->
        when (item.itemId) {
            R.id.navigation_home -> {
                updateView(R.id.navigation_home)
                return@OnNavigationItemSelectedListener true
            }
            R.id.navigation_dashboard -> {
                updateView(R.id.navigation_dashboard)
                return@OnNavigationItemSelectedListener true
            }
            R.id.navigation_notifications -> {
                updateView(R.id.navigation_notifications)
                return@OnNavigationItemSelectedListener true
            }
        }
        false
    }
    
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        showTabWithoutAddingToBackStack(fragmentA)
    
        navigation.setOnNavigationItemSelectedListener(onNavigationItemClicked)
    
    }
    
    private fun showTab(fragment: Fragment) {
        supportFragmentManager
                .beginTransaction()
                .replace(R.id.main_container, fragment, fragment::class.java.simpleName)
                .addToBackStack(fragment::class.java.simpleName)
                .commit()
    }
    
    fun showTabWithoutAddingToBackStack(fragment: Fragment) {
        supportFragmentManager
                .beginTransaction()
                .replace(R.id.main_container, fragment, fragment::class.java.simpleName)
                .commit()
    }
    
    fun setBottomTab(id: Int) {
        navigation.setOnNavigationItemSelectedListener(null)
        navigation.selectedItemId = id
        // currentTab = id
        navigation.setOnNavigationItemSelectedListener(onNavigationItemClicked)
    }
    }
    
  3. 创建 3 个新类,FragmentA、FragmentB 和 FragmentC:

    class FragmentA : Fragment() {
    
    override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?): View? {
        setHasOptionsMenu(true)
        return inflater.inflate(R.layout.fragment_a, container, false)
    }
    
    override fun onResume() {
        super.onResume()
        val act = activity as MainActivity
        act.setBottomTab(R.id.navigation_home)
    }
    }
    

使用这个 xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Fragment A" />
</LinearLayout>
  1. 启动应用程序
  2. 按“仪表板”- 显示片段 B
  3. 按“通知”- 显示片段 C
  4. 按“仪表板”- 显示片段 B
  5. 按“主页”- 显示片段 A
  6. 按返回按钮 - 显示片段 B
  7. 按返回按钮 - 应显示片段 C - 应用程序崩溃
  8. 按返回按钮 - 应显示片段 A - 应用程序崩溃
  9. 按返回按钮 - 应用程序关闭。 - 应用程序崩溃

Here is a video that demonstrates above steps

堆栈跟踪:

12-06 12:58:35.899 25903-25903/com.example.jimclermonts.bottomnavigationview E/InputEventSender: Exception dispatching finished signal.
12-06 12:58:35.900 25903-25903/com.example.jimclermonts.bottomnavigationview E/MessageQueue-JNI: Exception in MessageQueue callback: handleReceiveCallback
12-06 12:58:35.912 25903-25903/com.example.jimclermonts.bottomnavigationview E/MessageQueue-JNI: java.lang.**IllegalStateException: Fragment already added**: FragmentB{3aac1d9 #1 id=0x7f080059 FragmentB}
                                                                                                     at android.support.v4.app.FragmentManagerImpl.addFragment(FragmentManager.java:1882)
                                                                                                     at android.support.v4.app.BackStackRecord.executePopOps(BackStackRecord.java:825)
                                                                                                     at android.support.v4.app.FragmentManagerImpl.executeOps(FragmentManager.java:2577)
                                                                                                     at android.support.v4.app.FragmentManagerImpl.executeOpsTogether(FragmentManager.java:2367)
                                                                                                     at android.support.v4.app.FragmentManagerImpl.removeRedundantOperationsAndExecute(FragmentManager.java:2322)
                                                                                                     at android.support.v4.app.FragmentManagerImpl.popBackStackImmediate(FragmentManager.java:851)
                                                                                                     at android.support.v4.app.FragmentManagerImpl.popBackStackImmediate(FragmentManager.java:794)
                                                                                                     at android.support.v4.app.FragmentActivity.onBackPressed(FragmentActivity.java:174)

【问题讨论】:

  • “我想要 XXX 行为”好的,您当前的实现有什么问题?
  • @azizbekian 我希望它与 Youtube 应用程序相同。但是现在后退按钮的行为不一样了。
  • @JimClermonts “不一样”到底是什么意思?
  • 你的意思是你想喜欢这个:github.com/pedrovgs/DraggablePanel
  • @saif 我说的是底部导航,标签。它必须与 youtube 应用中的相同

标签: android bottomnavigationview


【解决方案1】:

我已经使用 Bottombar 库实现了这个概念。我已经上传到 GitHub。如有任何问题,请在此处查看并发表评论。

https://github.com/itvignes09/youtube-like-bttom-menu

样本输出

【讨论】:

  • 这个解决方案几乎 100% 正确。但是当 A,B,C,D,E,D,C,B,A 然后返回 5 次时,最后一次应该始终是 Home Fragment 然后它应该关闭。
  • 我已经更新了解决方案。请检查并告诉我
  • @ClassA YouTube 可以使用ViewPager 的子类,它禁用滑动并使用自定义PageTransformer,覆盖默认的幻灯片动画。考虑到 YouTube 应用不是开源的,我们无法确定。
  • 如何在像你这样的任何标签下添加子片段并将其维护在后台堆栈中?上面的例子如果没问题,但你可以用任何选项卡下的一个子片段更新它吗? @VigneswaranA
【解决方案2】:

这使您可以使用自定义后台堆栈(双端队列)实现“不要在后台堆栈上重复(但重新排序)我的片段”行为:

class MainActivity extends AppCompatActivity {
    private BottomNavigationView navigation;
    // initialize with number of different fragments
    private Deque<Integer> fragmentIds = new ArrayDeque<>(3);

    private FragmentA fragmentA = new FragmentA();
    private FragmentB fragmentB = new FragmentB();
    private FragmentC fragmentC = new FragmentC();

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        fragmentIds.push(R.id.navigation_home);
        showTabWithoutAddingToBackStack(fragmentA);
        navigation = findViewById(R.id.navigation);
        navigation.setOnNavigationItemSelectedListener(onNavigationItemClicked);
    }

    private BottomNavigationView.OnNavigationItemSelectedListener onNavigationItemClicked = new BottomNavigationView.OnNavigationItemSelectedListener() {
        @Override
        public boolean onNavigationItemSelected(@NonNull MenuItem item) {
            item.setChecked(true);
            int itemId = item.getItemId();
            if (fragmentIds.contains(itemId)) {
                fragmentIds.remove(itemId);
            }
            fragmentIds.push(itemId);
            showTabWithoutAddingToBackStack(getFragment(item.getItemId()));
            return true;
        }
    };

    private Fragment getFragment(int fragmentId) {
        switch (fragmentId) {
            case R.id.navigation_home:
                return fragmentA;
            case R.id.navigation_dashboard:
                return fragmentB;
            case R.id.navigation_notifications:
                return fragmentC;
        }
        return fragmentA;
    }

    void showTabWithoutAddingToBackStack(Fragment fragment) {
        getSupportFragmentManager().beginTransaction().replace(R.id.container, fragment, fragment.getClass().getSimpleName()).commit();
    }

    void setBottomTab(int id) {
        int itemIndex;
        switch (id) {
            case R.id.navigation_dashboard:
                itemIndex = 1;
                break;
            case R.id.navigation_notifications:
                itemIndex = 2;
                break;
            default:
            case R.id.navigation_home:
                itemIndex = 0;
        }
        navigation.getMenu().getItem(itemIndex).setChecked(true);
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) {
            onBackPressed();
            return true;
        }
        return super.onKeyDown(keyCode, event);
    }

    @Override
    public void onBackPressed() {
        fragmentIds.pop();
        if (!fragmentIds.isEmpty()) {
            showTabWithoutAddingToBackStack(getFragment(fragmentIds.peek()));
        } else {
            finish();
        }
    }
}

【讨论】:

  • 这个实现不正确。在 2 个标签之间快速按下时。回去时我必须按相同的次数。 Youtube 应用程序的 backstack 行为是它会记住 1 次后按的片段。
  • 其次,按“Home”、“Dashboard”、“Notifications”、“Dashboard”、“Home”时。后退按钮的行为应该是:“仪表板”、“通知”、“主页”、关闭应用程序。现在这不是它记住整个堆栈的情况。
  • 嗯,你不应该在你的问题中描述这种行为吗?用您自己的话来说,而不主要是指一个例子或“那里的一些应用程序”?类似于:“替换一个片段应该将被替换的片段添加到后台堆栈,只有当该片段之前没有被添加到后台堆栈或者已经从后台堆栈中弹出时。”相反,我们收到一条我无法用您的代码重现的错误消息?
  • @JimClermonts 请检查我的新实现。如果不符合您的要求,请公开!
【解决方案3】:

我必须对其进行一些更改并使其正常工作,并且我已经在多个设备上进行了测试,并且在一次测试运行中切换了 100 多次选项卡。 它工作正常,更改以下代码,

private fun updateView(fragmentId: Int) {
    var exists = false
    fragmentIds
            .filter { it == fragmentId }
            .forEach { exists = true }

    if (exists) {
        fragmentIds.remove(fragmentId)
        showTabWithoutAddingToBackStack(getFragment(fragmentId))
    } else {
        fragmentIds.add(fragmentId)
        showTab(getFragment(fragmentId))
    }
}

对于这个新代码,

private fun updateView(fragmentId: Int) {
    var exists = false
    fragmentIds
            .filter { it == fragmentId }
            .forEach { exists = true }

    if (exists) {
        showTab(getFragment(fragmentId))
        setBottomTab(fragmentId)
    } else {
        fragmentIds.add(fragmentId)
        showTab(getFragment(fragmentId))
    }
}

就是这样!!!

【讨论】:

    【解决方案4】:

    您好,请检查一下我做了这个并且工作正常。 你可以检查一下。 https://github.com/sandeshsk/BackStackFragmentRedirectsToHome

    如果有任何问题,请更新。

    这是一个分配片段的方法

    public void addFragment(FragmentManager fragmentManager,
                                   Fragment fragment,
                                   int containerId,boolean isFromHome){
    
        fragmentManager.popBackStack(null,FragmentManager.POP_BACK_STACK_INCLUSIVE);
    
        FragmentTransaction fragmentTransaction=fragmentManager.beginTransaction();
        if(isFromHome){
            fragmentTransaction.replace(containerId,fragment);
        }else{
            fragmentTransaction.add(new HomeFragment(),"Home");
            fragmentTransaction.addToBackStack("Home");
        }
        fragmentTransaction.replace(containerId,fragment).commit();
    
    }
    

    这是您的导航项监听器

     private BottomNavigationView.OnNavigationItemSelectedListener mOnNavigationItemSelectedListener
            = new BottomNavigationView.OnNavigationItemSelectedListener() {
    
        @Override
        public boolean onNavigationItemSelected(@NonNull MenuItem item) {
            switch (item.getItemId()) {
                case R.id.navigation_home:
                  if (getSupportFragmentManager().getBackStackEntryCount() == 0) {
                        addFragment(getSupportFragmentManager(), new HomeFragment(), R.id.frame, true);
                    }else{
                        getSupportFragmentManager().popBackStack();
                    }
                    return true;
                case R.id.navigation_dashboard:
                    addFragment(getSupportFragmentManager(),new DashboardFragment(),R.id.frame,false);
                    return true;
                case R.id.navigation_notifications:
                    addFragment(getSupportFragmentManager(),new NotificationFragment(),R.id.frame,false);
                    return true;
                case R.id.navigation_setting:
                    addFragment(getSupportFragmentManager(),new SettingFragment(),R.id.frame,false);
                    return true;
            }
            return false;
        }
    };
    

    onBackPressed 方法

     @Override
    public void onBackPressed() {
        if(getSupportFragmentManager().getBackStackEntryCount()>0){
            navigation.setSelectedItemId(R.id.navigation_home);
        }else {
            super.onBackPressed();
        }
    }
    

    【讨论】:

      【解决方案5】:

      如果你想像 Youtube 一样返回堆栈上的所有项目(不仅是 home),试试下面的代码:

      BottomNavigationView bottomNavigationView;
      FragmentTransaction transaction;
      
      @Override
      protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
          setContentView(R.layout.activity_main);
          bottomNavigationView = (BottomNavigationView) findViewById(R.id.navigation);
      
          bottomNavigationView.setOnNavigationItemSelectedListener
                  (new BottomNavigationView.OnNavigationItemSelectedListener() {
                      @Override
                      public boolean onNavigationItemSelected(@NonNull MenuItem item) {
                          Fragment selectedFragment = null;
                          switch (item.getItemId()) {
                              case R.id.action_item1:
                                  selectedFragment = ItemOneFragment.newInstance();
                                  break;
                              case R.id.action_item2:
                                  selectedFragment = ItemTwoFragment.newInstance();
                                  break;
                              case R.id.action_item3:
                                  selectedFragment = ItemThreeFragment.newInstance();
                                  break;
                              case R.id.action_item4:
                                  selectedFragment = ItemFourFragment.newInstance();
                                  break;
                              case R.id.action_item5:
                                  selectedFragment = ItemFiveFragment.newInstance();
                                  break;
                          }
                          transaction = getSupportFragmentManager().beginTransaction();
                          transaction.replace(R.id.frame_layout, selectedFragment);
                          transaction.addToBackStack(null);
                          transaction.commit();
                          return true;
                      }
                  });
      
          //Manually displaying the first fragment - one time only
          transaction = getSupportFragmentManager().beginTransaction();
          transaction.replace(R.id.frame_layout, ItemOneFragment.newInstance());
          transaction.commit();
      
      }
      
      @Override
      public void onBackPressed() {
          int count = getSupportFragmentManager().getBackStackEntryCount();
          if (count == 0) {
              super.onBackPressed();
          } else {
              int index = ((getSupportFragmentManager().getBackStackEntryCount()) -1);
              getSupportFragmentManager().popBackStack();
              FragmentManager.BackStackEntry backEntry = getSupportFragmentManager().getBackStackEntryAt(index);
              int stackId = backEntry.getId();
              bottomNavigationView.getMenu().getItem(stackId).setChecked(true);
          }
      }
      

      【讨论】:

      • 随机切换标签然后按回时崩溃
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2013-09-17
      • 2020-04-13
      • 1970-01-01
      • 2020-08-02
      • 1970-01-01
      • 2013-04-02
      • 1970-01-01
      相关资源
      最近更新 更多