如何解决使用NavigationUI切换回片段时调用OnDestory
我正在将NavigationUI与底部导航视图一起使用,并且还通过操作栏对其进行了设置。
我的应用以首页片段开始,当我切换到搜索片段OnDestroyView()
时,homeFragment
被调用(预期),但是当我从搜索片段切换回首页片段{{ 1}}被调用,然后显示onDestroy()
的新实例,而不是它在我留给它的状态下恢复的新实例。如何使它恢复到以前的状态?
这是我设置NavigationUI的方式:
homefragment
我的导航图:
NavController navController = Navigation.findNavController(this,R.id.navigation_host_fragment);
NavigationUI.setupWithNavController(mainBinding.mainActivityBn,navController);
NavigationUI.setupActionBarWithNavController(this,navController);
编辑:我正在使用底部导航视图在片段之间导航。当我按下“后退”按钮时,将加载原始片段,就像以前的状态一样(预期行为)。
解决方法
根据this issue:
要了解此处的逻辑,我们需要查看选择目的地时
NavigationUI
的作用。查看the source code并删除不相关的代码(动画,辅助菜单代码):NavOptions.Builder builder = new NavOptions.Builder() .setLaunchSingleTop(true); builder.setPopUpTo(findStartDestination(navController.getGraph()).getId(),false); NavOptions options = builder.build(); navController.navigate(item.getItemId(),null,options);
因此,
popUpTo
会弹出到根目标,但使用false
不会弹出根目标本身。然后,它使用setLaunchSingleTop(true)
来避免在后堆栈上具有两个起始目标实例。这样做的目的是在返回根目录目标时,不清除与该根目录目标相关的任何ViewModel
或SavedStateHandle
。因此NavController调用
然后navigate()
,我们将所有内容弹出(但不包括起始目的地)(不包括您的第一个屏幕)从后堆栈中弹出,从而正确删除了第二个屏幕。NavController调用到
FragmentNavigator
以进行navigate()
调用。FragmentNavigator
看到launchSingleTop
是正确的,并正确地确定该片段已经在堆栈的顶部。但是,片段与活动不同,对于模仿onNewIntent()
的行为,onNewArguments()
没有等同于singleTop
的{{1}},因此FragmentNavigator
会弹出先前的片段并添加一个新的片段,传入新的参数(此处为null
)。
因此,由于片段如何处理setLaunchSingleTop(true)
标志,因此单击底部导航栏中的第一项会破坏并再次创建片段。这就是系统后退按钮行为不同的原因-系统后退按钮仅调用popBackStack()
,而不使用任何单个top行为。
该问题仍然存在,并表明:
FragmentNavigator
具有单个顶部的行为很糟糕,应该感到很糟糕。它与活动的行为不匹配并且是不直观的。如果您不介意,让我们使用此错误来跟踪工作。理想情况下,将创建 not 的新实例,这意味着将仅恢复原始MapFragment,并将其放回原处。这将与您按下系统后退按钮时发生的情况一致。
没有直接解决此问题的方法,因此,最好的结果(未发生此情况)将要求您给该问题加注标,然后等待它解决。但是,您会注意到,保存在NavBackStackEntry 中的任何状态或ViewModel都会通过singleTop
操作进行保存:
// Instead of using this:
myViewModel = new ViewModelProvider(this).get(MyViewModel.class);
// You could use:
NavController navController = NavHostFragment.findNavController(this);
NavBackStackEntry entry = navController.getBackStackEntry(R.id.homeFragment);
myViewModel = new ViewModelProvider(entry).get(MyViewModel.class);
但是,这无法保存Fragment的视图状态(即滚动位置),因为无法将其保存到NavBackStackEntry中,而只能保存到NavBackStackEntry中 保存在“片段”级别。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。