将ViewPager内的ListFragment替换为Fragment与选项卡。

4
我试图在我的应用程序中设置以下导航:
实际情况:ViewPager + 选项卡可以在列表之间滑动:
- ListFragment A - ListFragment B
期望的情况:ViewPager + 选项卡:
- 当选择ListFragment A的列表项时,用DetailFragment A替换ListFragment A。 - 当选择ListFragment B的列表项时,用DetailFragment B替换ListFragment B。
目标是在选项卡导航中显示详细片段。我无法用详细片段替换fragmentList(fragmentList没有自定义布局,因此没有ID,我不知道如何做)。此外,启动新活动会隐藏选项卡栏。请问有人能帮帮我吗?
1个回答

14

我会创建一个包装片段(fragment),该片段了解要显示的内容 - 列表或细节。这将完全与 ViewPager 解耦 - ViewPager 只知道它持有包装片段,而这些片段将自己管理其内容。

实现方式如下:

public class WrapperFragment extends Fragment {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        MyListFragment list = new MyListFragment();
        list.setListAdapter(adapter);
        list.setOnItemClickListener(new OnItemClickListener() {
            @Override public void onItemClick(AdapterView<?> l, View v, int position, long id) {
              // Create details fragment based on clicked item's position
              Fragment details = new Fragment();
              getChildFragmentManager()
                  .beginTransaction()
                  .replace(R.id.container, details)
                  .addToBackStack(null)
                  .commit();
            }
        });

        getChildFragmentManager()
            .beginTransaction()
            .add(R.id.container, list)
            .commit();
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        // There has to be a view with id `container` inside `wrapper.xml`
        return inflater.inflate(R.layout.wrapper, container, false);
    }

    public class MyListFragment extends ListFragment {

        private OnItemClickListener listener;

        public void setOnItemClickListener(OnItemClickListener l) {
            this.listener = l;
        }

        @Override
        public void onListItemClick(ListView l, View v, int position, long id) {
            if(listener != null) {
              listener.onItemClick(l, v, position, id);
            }
        }
    }
}

wrapper.xml的想法是,WrapperFragment必须提供一个包含具有id container 的视图的布局 - 因为我们正在使用该id的视图来放置子片段 - MyListFragmentDetailsFragment

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <FrameLayout
        android:id="@+id/container"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</FrameLayout>

另一种方法。这应该可以工作,但是你需要尝试一下(布局本身有id,而不是具有container ID的子元素):

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

1
已经添加了 wrapper.xml 的示例。如果您在处理此问题时遇到困难,请告诉我。 - mindeh
1
getChildFragmentManager().popBackStack()。注意在显示 DetailsFragment 时的 .addToBackStack(null) 行。它基本上是说“将此事务保存为历史操作 - 可以通过按 HW 返回按钮来返回的操作之一”。而 popBackStack() 类似于模拟 HW 返回按钮点击。将 backStack 管理视为类似于 HTML5 历史记录 API,可能更容易理解。 - mindeh
1
首先感谢这个,它解决了我过去几天一直遇到的问题。其次,关于后退栈的问题,我发现将getChildFragmentManager()更改为来自support.app的常规FragmentManager()使得后退按钮可以在任何地方自动调用而无需手动调用,使用childmanager则不行。不确定这会有什么副作用,但它的行为方式符合我的期望:D - fury-s12
+1包装器很好用!但是你知道为什么onCreateOptionsMenu不再起作用了吗?我只需要它来处理我的listFragment。 - Kevin van Mierlo
1
@Kevin,抱歉,错过了你的评论。原因可能是onCreateOptionsMenu()仅在顶级片段-包装器上调用。解决方案可以从那里创建菜单-例如通过手动调用listFragment#onCreateOptionsMenu()。这只是我想到的,你需要尝试一下。 - mindeh
显示剩余5条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接