【问题标题】:Replace, add, replace fragment, then the Navigation Back button not working for first fragment替换,添加,替换片段,然后导航返回按钮不适用于第一个片段
【发布时间】:2018-08-03 02:31:53
【问题描述】:

我有一个片段堆栈,我在其中一起使用替换和添加。添加或替换我的片段的代码(在我的活动中)如下

private fun addFragment(fragment: Fragment, name: String) {
    supportFragmentManager.beginTransaction().add(R.id.container, fragment)
            .addToBackStack(name).commit()
}

private fun replaceFragment(fragment: Fragment, name: String) {
    supportFragmentManager.beginTransaction().replace(R.id.container, fragment)
            .addToBackStack(name).commit()
}

在我的片段中,我确实有一个带有主菜单后退图标的工具栏。单击后,它应该有助于将我的片段弹出到上一个堆栈。

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
    super.onViewCreated(view, savedInstanceState)
    toolbar_actionbar.setNavigationIcon(R.drawable.ic_arrow_back_black_24dp)
    setHasOptionsMenu(true)
    (activity as AppCompatActivity).setSupportActionBar(toolbar_actionbar)
}

override fun onOptionsItemSelected(item: MenuItem): Boolean {
    return if (item.itemId == android.R.id.home) {
        activity?.onBackPressed()
        true
    } else {
        super.onOptionsItemSelected(item)
    }
}

为了清楚起见,我显示活动 onBackPressed 编码如下

override fun onBackPressed() {
    if (supportFragmentManager.backStackEntryCount > 0) {
        supportFragmentManager.popBackStackImmediate()
    } else {
        super.onBackPressed()
    }
}

现在如果我添加fragment1,添加fragment2,添加fragment3,添加fragment4,然后按返回,返回,返回,返回......一切正常。

同样,如果我用fragment1替换,用fragment2替换,用fragment3替换,用fragment4替换,然后返回,返回,返回,返回,一切正常。

但是,如果我这样做了,请用片段 1 替换,添加片段 2,然后用片段 3 替换,然后按返回、返回、返回...。第三次按返回不再起作用。为什么??

为了更好地说明这一点,我将代码放在 github 中,如下所示 https://github.com/elye/issue_android_fragment_replace_add_replace

并录制成下面的gif(63秒gif)显示,4个添加作品,4个替换作品,但是替换和添加的混合,会导致工具栏后退按钮在几次返回后不起作用。

注意添加片段显示重叠数,因为背景是透明的。这样做是为了轻松区分添加与替换。

我怀疑这是一个谷歌错误,但如果我错过任何重要的事情,应该分享一下。

【问题讨论】:

标签: android android-fragments


【解决方案1】:

经过调查,问题似乎如下所述。

因为每个片段都有自己的操作栏集(在此 stackoverflow 中,我将其简化为仅单击 R.id.home 以更轻松地说明问题)。添加/删除片段时,每个都会调用以下代码。

(activity as AppCompatActivity).setSupportActionBar(toolbar_actionbar)

当我们执行 add-add-replace(或 replace-add-replace)时......我们在 backstack 中有 3 个片段,但只有一个可见。因为最后的replace,会移除之前的2个fragment。

当我们点击返回按钮时,会弹出最后一个被替换的片段,然后系统恢复前2个片段。

在恢复前 2 个片段期间,会在短时间内为两个片段调用以下代码

(activity as AppCompatActivity).setSupportActionBar(toolbar_actionbar)

我怀疑这会导致第一个片段的工具栏未完全设置的某些行为(Android SDK 错误?)。

为了解决这个问题,当我们再次弹出第二个片段时,我们必须强制第一个片段调用

(作为 AppCompatActivity 的活动).setSupportActionBar(toolbar_actionbar)

对于上述问题,我的解决方法是将上面的代码放在 onResume() 中。

override fun onResume() {
    super.onResume()
    (activity as AppCompatActivity).setSupportActionBar(toolbar_actionbar)
}

而且每当一个fragment backstack改变时,我都会调用顶部fragment的onResume

    supportFragmentManager.addOnBackStackChangedListener {
        val currentFragment = supportFragmentManager.findFragmentById(R.id.container)
        currentFragment?.onResume()
    }

这现在有助于确保无论替换-添加混合堆栈,工具栏菜单都将继续工作。

我有示例解决方法代码 https://github.com/elye/issue_android_fragment_replace_add_replace_workaround

这确实是一种解决方法而不是解决方案。我希望发布一些更好的答案,或者如果这确实是一个 Google Bug,可以在以后的 SDK 版本中对其进行修复。

【讨论】:

【解决方案2】:

试试这个

if (supportFragmentManager.backStackEntryCount > 1) {
        supportFragmentManager.popBackStackImmediate()
    } else {
        super.onBackPressed()
    }

【讨论】:

  • 谢谢。但这不是重点。问题是 onOptionsItemSelected 在某些时候不再被触发。
【解决方案3】:

您好,将您的 工具栏代码及其 xml 部分放入 MainActivity 及其相关布局从所有 fargment 中删除 及其布局。 并输入如下代码:

override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        toolbar_actionbar.setNavigationIcon(R.drawable.ic_arrow_back_black_24dp)
        setSupportActionBar(toolbar_actionbar)
}

并将下面的代码放入 MainActivity 以解决它。

override fun onOptionsItemSelected(item: MenuItem): Boolean {
        return if (item.itemId == android.R.id.home) {
            if (supportFragmentManager.backStackEntryCount > 0) {
                supportFragmentManager.popBackStackImmediate()
            } else {
                super.onBackPressed()
            }
            true
        } else {
            super.onOptionsItemSelected(item)
        }
    }

我已经测试过并且工作正常。希望它可以帮助你。 如果您有任何疑问,请在下面发表评论/留言,我会向您解释并解决您的问题。

【讨论】:

  • 谢谢。我真的不能那样做,因为在实际项目中,每个片段都有自己的菜单项,而不是主页按钮。所以每个片段都需要有它的工具栏。这里提供的是奇怪行为的简化版本。
  • 谢谢。希望你有一些好的发现。我刚刚发现 Add-Add-Replace 也有同样的问题。这个想法是一旦你添加两次......然后替换。当您从替换中返回时,前两个片段会同时恢复。当这种情况发生时,只有一个菜单项被激活触发。我认为问题可能是这样的。
猜你喜欢
  • 1970-01-01
  • 2017-08-03
  • 2021-11-30
  • 1970-01-01
  • 2019-10-02
  • 1970-01-01
  • 1970-01-01
  • 2012-04-21
  • 1970-01-01
相关资源
最近更新 更多