Nexus 6设备和仿真器

时间:2015-06-19 01:40:40

标签: android android-fragments fragment-backstack

我正在处理一个支持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,并且它们都没有其他用户可以访问的片段或活动。

  • HomeFragment
  • CalendarFragment
  • StandingsFragment
  • PlayerSearchFragment(见下文)
  • AboutFragment
  • FeedbackFragment

这些片段功能的唯一例外是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();
}

我希望所有设备的体验都是相同的,无论是硬件还是模拟器。

3 个答案:

答案 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仿真器中所需的功能。