我正在处理一个支持Fragment
并带有SearchView
窗口小部件和RecyclerView
的应用,以显示搜索结果,将用户发送到FragmentActivity
以显示选择的细节。所有这一切都很好,但我看到Nexus 6仿真器和实际设备之间在后台堆栈方面的行为不一致。在模拟器中,一切都按照我的预期工作,如果在细节FragmentActivity上按下Back按钮,用户将被带回搜索结果片段。在实际的Nexus 6设备上,用户将一直回到包含我的应用程序菜单的AppCompatActivity
(此活动使用支持FragmentManager
,它将管理的片段添加到后台堆栈中:
private void replaceFragment(Fragment supportFragment) {
String fragmentName = supportFragment.getClass().getSimpleName();
supportFragmentManager.beginTransaction()
.addToBackStack(fragmentName)
.replace(R.id.frame_menu_container, supportFragment, fragmentName)
.commit();
}
将用户从Fragment发送到FragmentActivity的代码只是一个额外的意图:
Intent intent = new Intent(getActivity(), PlayerDetailsActivity.class);
intent.putExtra(TransientPlayer.BUNDLE_KEY, transientPlayer);
startActivity(intent);
我对backstack没有做任何特别的事情(还有) - 这是默认行为。
这可能是什么问题?我已经在backstack上做了一些阅读,当你从FragmentActivy返回到你没有使用FragmentManager的片段时,我还没有看到任何关于手动向backstack添加内容的东西。
编辑1:这是定义frame_menu_container
的布局XML:
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:gravity="center"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="?attr/colorPrimary"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
app:popupTheme="@style/ThemeOverlay.AppCompat.Light"/>
<FrameLayout
android:id="@+id/frame_menu_container"
android:layout_below="@id/toolbar"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:layout="@layout/fragment_home">
</FrameLayout>
</RelativeLayout>
编辑2:好的,这是我的高级活动和片段的文字描述:
NavigationMenuActivity
,扩展了AppCompatActivity
。这是根据需要交换所有片段的地方,使用SupportFragmentManager
并使用replaceFragment()
方法。
我有以下片段,所有片段都显示静态信息,进行REST调用以检索和显示数据,或允许用户发送反馈。全部扩展android.support.v4.app.Fragment
,并且它们都没有其他用户可以访问的片段或活动。
这些片段功能的唯一例外是PlayerSearchFragment
,它也扩展了android.support.v4.app.Fragment
。此片段包含RecyclerView
,用于在用户想要搜索播放器时显示REST调用的结果。当返回结果并且用户从列表中选择一个项目时,它们将被发送到PlayerDetailsActivity
,其扩展为FragmentActivity
。
PlayerDetailsActivity
使用FragmentTabHost
视图,其中包含有关可以查看的播放器的不同类型的信息。这是该路径的“行尾” - 用户无法访问其他片段或活动。这就是问题所在。当我在此活动中点击“后退”按钮时,我的目的是让用户返回PlayerSearchFragment
片段(搜索结果),在这种情况下,如果我在Nexus 6 模拟器,但如果我使用的是实际的Nexus 6 设备,它们会一直回到NavigationMenuActivity
活动。
我在按下后退按钮时调用的PlayerDetailsActivity
中有以下方法,但它始终显示零条目:
@Override
public void onBackPressed() {
FragmentManager fragmentManager = getSupportFragmentManager();
int count = fragmentManager.getBackStackEntryCount();
Log.i(TAG, "Backstack : There are " + count + " entries");
for (int i = 0; i > count; i++) {
FragmentManager.BackStackEntry backStackEntry = fragmentManager.getBackStackEntryAt(i);
Log.i(TAG, String.format("Backstack : id=%d, name=%s, shortTitle=%s, breadcrumbTitle=%s",
backStackEntry.getId(),
backStackEntry.getName(),
backStackEntry.getBreadCrumbShortTitle(),
backStackEntry.getBreadCrumbTitle()));
}
super.onBackPressed();
}
我希望所有设备的体验都是相同的,无论是硬件还是模拟器。
答案 0 :(得分:1)
首先根据您的方案,您不需要向BackStack添加交易,因为它仅在您在同一活动中的片段之间导航时使用。
现在再说一点,当你再次按下时,你没有得到你的片段是你点击了两次。单击一次删除您的第二个活动,另一个删除在您的Backstack中添加的片段。至于你的信息Fragment没有backpress回调,它们依赖于onbackPressed()方法的活动
我尝试运行此代码,并且它在我的Nexus 5设备上正常运行。 虽然我附上了我正在使用的代码
public class MainActivity extends FragmentActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
replaceFragment(new FragmentA());
}
private void replaceFragment(Fragment supportFragment) {
getSupportFragmentManager().beginTransaction()
.replace(R.id.frame_menu_container, supportFragment
.commit();
}
}
FragmentA在这里:
public class FragmentA extends Fragment {
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_a, null);
view.findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(getActivity(), AnotherActivity.class);
startActivity(intent);
}
});
return view;
}
}
另一项活动就像:
public class AnotherActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.another_act);
}
}
我认为你不应该遇到任何问题。
如果您仍然遇到问题,请告诉我您的设备名称并在此处附上代码,以便我可以测试并告知您。
答案 1 :(得分:1)
好好试试这个,希望能让你摆脱这个漏洞。
这个答案为您提供了解决方法,但没有给出真正发生的事情
@Override
public void onBackPressed() {
FragmentManager fragmentManager = getSupportFragmentManager();
int count = fragmentManager.getBackStackEntryCount();
Log.i(TAG, "Backstack : There are " + count + " entries");
if(count > 0){
fragmentManager.popBackStack(); //or popBackStackImmediate();
for (int i = 0; i > count; i++) {
FragmentManager.BackStackEntry backStackEntry = fragmentManager.getBackStackEntryAt(i);
Log.i(TAG, String.format("Backstack : id=%d, name=%s, shortTitle=%s, breadcrumbTitle=%s",
backStackEntry.getId(),
backStackEntry.getName(),
backStackEntry.getBreadCrumbShortTitle(),
backStackEntry.getBreadCrumbTitle()));
}
}else{
super.onBackPressed();
}
}
答案 2 :(得分:1)
虽然我仍然没有回答为什么我不能按照我希望的方式工作,但我确实有一个解决方法,它确实适用于我需要做的事情。基本上,我覆盖PlayerDetailsActivity
中的向上和向后导航回调,并使用intent(带数据)将用户显式发送回NavigationMenuActivity
。我在意图中发送的数据基本上只是一个布尔值,它告诉我是否来自PlayerDetailsActivity
,如果是,则显示该片段而不是Home片段,如下所示:
@Override
public void onBackPressed() {
sendBackToMenu();
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
sendBackToMenu();
finish();
return true;
}
return false;
}
private void sendBackToMenu() {
Intent backToMenuIntent = new Intent(this, NavigationMenuActivity.class);
backToMenuIntent.putExtra("FROM_PLAYER_DETAILS", true);
startActivity(backToMenuIntent);
}
以上更改是在我将类更改为从AppCompatActivity
扩展并在PlayerDetailsActivity.onCreate()
中添加以下内容以激活ActionBar for Up导航之后:
ActionBar actionBar = getSupportActionBar();
assert actionBar != null;
actionBar.setTitle(R.string.player_details);
actionBar.setDisplayHomeAsUpEnabled(true);
另外,在NavigationMenuActivity.onCreate()
我这样做:
if (getIntent().getBooleanExtra("FROM_PLAYER_DETAILS", false))
replaceFragment(new PlayerSearchFragment());
else
replaceFragment(new HomeFragment());
我不知道这是否是最合适的方式,但它可以很好地完成我在少数硬件设备和Nexus 4/5/6仿真器中所需的功能。