4

在内存中,我有一个项目列表,每个项目都有一个标题和一个 0..N 个块的列表(存储一些信息)。我知道我在内存中正确地创建了这个列表,因为我已经将整个结构输出到了命令行。

我有一个使用 ViewPager 来显示这些数据的活动。ViewPager 的每一页都是一个 Item,它充满了 Fragment,这些 Fragments 保存了 Item 的每个 Chunk 的信息。

按照我目前的编码方式,前两个项目的块显示在第一个项目下的第一页上。当页面滚动到第二页时,它是空的。当页面滚动到第 3 页时(我现在只有 3 个项目的列表供测试)它也显示一个空白页,但是,当页面返回到第 2 页时,它显示第 1 个项目的块. 然后,每次导航到第一页然后是第二页,第三页的块被新添加到现有列表中。如果有从第 2 页到第 3 页的导航并返回,则第 1 项的块将新添加到列表中。这可以通过不断增长的列表来完成。我不知道为什么它的行为不正常。

这是我的主要活动(包括 FragmentPagerAdapter):

public class MyActivity extends FragmentActivity {
    MyPagerAdapter mPagerAdapter;
    ViewPager mViewPager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Controller.setup();

        mPagerAdapter = new MyPagerAdapter(getSupportFragmentManager());

        mViewPager = (ViewPager) findViewById(R.id.pager);
        mViewPager.setAdapter(mPagerAdapter);
    }

    public class MyPagerAdapter extends FragmentPagerAdapter {
        public MyPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            Fragment fragment = new MyFragment();

            Bundle args = new Bundle();
            args.putInt(MyFragment.POSITION, position);
            fragment.setArguments(args);

            return fragment;
        }

        @Override
        public int getCount(){
            return Controller.getAmount();
        }

        @Override
        public CharSequence getPageTitle(int position) {
            return Controller.getItem(position).getTitle();
        }
    }
}

Controller 非常简单,并按照您的预期返回信息。Controller.setup() 只是创建项目列表。

这是我的项目片段:

public class MyFragment extends Fragment {
    public final static String POSITION = "position";
    private int pos;

    @Override
    public void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        pos = getArguments().getInt(POSITION);
    }
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        ViewGroup rootView = (ViewGroup) inflater.inflate(R.layout.fragment_item, container, false);

        Item item = Controller.getItem(pos);
        Iterator<Chunk> chunks = item.getChunks().iterator();
        FragmentManager manager = getFragmentManager();
        FragmentTransaction transaction = manager.beginTransaction();

        while(chunks.hasNext()){
            Chunk chunk = chunks.next();
            ChunkFragment entry = new ChunkFragment();

            Bundle args = new Bundle();         
            args.putString(ChunkFragment.INFO_ARG, chunk.getInfo());
            entry.setArguments(args);

            transaction.add(R.id.container, entry);
        }
        transaction.commit();

        return rootView;
    }
}

最后是我的 Chunk 片段:

public class ChunkFragment extends Fragment {
    public static final String INFO_ARG = "Info";
    private String mInfo;

    @Override
    public void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        Bundle args = getArguments();
        mInfo = args.getString(INFO_ARG);
    }
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        //setup view
        ViewGroup rootView;

        rootView = (ViewGroup) inflater.inflate(R.layout.fragment_chunk, container, false);

        TextView infoView = (TextView) rootView.findViewById(R.id.info_text);
        infoView.setText(mInfo);

        return rootView;
    }
}

我知道这是很多代码,但我不知道问题出在哪里。我已将其简化到最低限度,仅包括可能成为问题根源的操作。我也没有包含我的 XML 布局文件,但它们已经过测试并且工作正常。

4

1 回答 1

3

我对此不确定,但我相信您的问题可能与在 ViewPager 中的片段中实例化片段的困难有关。

您可以考虑通过遵循嵌套片段的指导方针将其取消资格作为一种可能性:这基本上归结为getChildFragmentManager()在要添加片段的片段上使用。

此外,您可能需要考虑扩展FragmentStatePagerAdapter,它可以更好地在它们的破坏和重新创建之间保持片段状态。

于 2013-04-03T19:28:12.863 回答