繁体   English   中英

Android 导航组件:如何保存片段状态

[英]Android navigation component: how save fragment state

我使用 bottomNavigationView 和导航组件。 请告诉我如何在切换到另一个选项卡并返回旧选项卡后不破坏片段? 例如,我有三个选项卡 - A、B、C。我的开始选项卡是 A。导航到 B 后,然后返回 A。当我返回选项卡 A 时,我不希望它被重新创建。 怎么办? 谢谢

根据open issue ,Navigation 不直接支持多个返回堆栈 - 即,当您从 A 或 C 返回到 B 时保存堆栈 B 的状态,因为 Fragment 不支持多个返回堆栈。

根据此评论

NavigationAdvancedSample 现在可在https://github.com/googlesamples/android-architecture-components/tree/master/NavigationAdvancedSample 获得

此示例使用多个 NavHostFragment,每个底部导航选项卡一个,以解决 Fragment API 在支持多个返回堆栈方面的当前限制。

我们将继续使用 Fragment API 以支持多个返回堆栈,并在创建后将 Navigation API 插入其中,这将消除对NavigationExtensions.kt文件等任何内容的需要。 我们将继续使用此问题来跟踪该工作。

因此,您现在可以在您的应用程序中使用 NavigationAdvancedSample 方法并为问题加星标,以便您在解决基础问题并向导航添加直接支持时获得更新。

如果您可以处理销毁片段,但想要保存 ViewModel,您可以将其范围限定到导航图中:

private val viewModel: FavouritesViewModel by 
    navGraphViewModels(R.id.mobile_navigation) {
        viewModelFactory
    }

在这里阅读更多

编辑

正如@SpiralDev 所指出的,使用 Hilt 简化了一点:

private val viewModel: MainViewModel by 
    navGraphViewModels(R.id.mobile_navigation) {
         defaultViewModelProviderFactory     
    }

只需使用导航组件版本 2.4.0-alpha01 或更高版本

class BaseViewModel : ViewModel() {

    val bundleFromFragment = MutableLiveData<Bundle>()
}


class HomeViewModel : BaseViewModel () {

   ... HomeViewModel logic
}

内部主页片段(底部导航选项卡)

private var viewModel: HomeViewModel by viewModels()

override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
    super.onViewCreated(view, savedInstanceState)

    viewModel.bundleFromFragment.observe(viewLifecycleOwner, Observer {
      
        val message = it.getString("ARGUMENT_MESSAGE", "")
       binding.edtName.text = message
    })
}

override fun onDestroyView() {
    super.onDestroyView()
    viewModel.bundleFromFragment.value = bundleOf(
        "ARGUMENT_MESSAGE" to binding.edtName.text.toString(),
        "SCROLL_POSITION" to binding.scrollable.scrollY
    )
}

您可以对底部导航中的所有片段执行此模式

更新 2021使用版本 2.4.0-alpha05 或更高版本。 不要使用这个答案或其他等。

这可以使用 Fragment 显示/隐藏逻辑来实现。

private val bottomFragmentMap = hashMapOf<Int, Fragment>()
bottomFragmentMap[0] = FragmentA.newInstance()
bottomFragmentMap[1] = FragmentB.newInstance()
bottomFragmentMap[2] = FragmentC.newInstance()
bottomFragmentMap[3] = FragmentD.newInstance()


private fun loadFragment(fragmentIndex: Int) {
    val fragmentTransaction = childFragmentManager.beginTransaction()

    val bottomFragment = bottomFragmentMap[fragmentIndex]!!

    // first time case. Add to container
    if (!bottomFragment.isAdded) {
        fragmentTransaction.add(R.id.container, bottomFragment)
    }

    // hide remaining fragments
    for ((key, value) in bottomFragmentMap) {
        if (key == fragmentIndex) {
            fragmentTransaction.show(value)
        } else if (value.isVisible) {
            fragmentTransaction.hide(value)
        }
    }
    fragmentTransaction.commit()
}

在活动上声明片段并在 onCreate 方法上创建片段实例,然后在 updateFragment 方法中传递片段实例。 根据需要创建与底部导航侦听器项 id 对应的尽可能多的片段实例。

Fragment fragmentA;
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_home);

fragmentA = new Fragment();
updateFragment(fragmentA);
}

public void updateFragment(Fragment fragment) {
FragmentTransaction transaction = 
getSupportFragmentManager().beginTransaction();
transaction.add(R.id.layoutFragment, fragment);
transaction.commit();
}

此外,请确保您正在使用 android.support.v4.app.Fragment 并调用 getSupportFragmentManager()

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM