ViewPager 在屏幕旋转后重新实例化无序的项目

2024-04-09

我正在使用一个ViewPager包含多个 ListView,其代码类似于答案中的代码无限ViewPager https://stackoverflow.com/questions/7440012/infinite-viewpager。这个想法是为谷歌日历应用程序提供类似日视图的东西(其来源似乎不可用;只有默认的日历应用程序是,但它使用ViewSwitcher) - 我想让用户看起来可以无限左右滑动,但实际上只有 3 个项目ViewPager,当用户点击页面 0 或 2 时,我们将 1 设置为当前页面并相应更新。

现在,这一切都有效了。然而,奇怪的是,当手机旋转并重建活动时(我避免使用configChanges目前),应用程序中的页面会再次实例化,但顺序不正确。顺序是 1->0->2,而不是 0->1->2,这会打乱应用程序中页面的顺序。

我的片段,在onActivityCreated():

mPagerAdapter = new ContinuousPagerAdapter(R.layout.my_listview, this);

// set the adapter
mViewPager = (ViewPager) getView().findViewById(R.id.agendaViewPager);
mViewPager.setOffscreenPageLimit(3);
mViewPager.setOnPageChangeListener(this);
mViewPager.setAdapter(mPagerAdapter);
mViewPager.setSaveEnabled(false);
// ...
mViewPager.setCurrentItem(1, false);
loadData();

寻呼机适配器:

import android.content.Context;
import android.os.Parcelable;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;

public class ContinuousPagerAdapter extends PagerAdapter {

    OnPageInstantiatedListener pListener;
    ViewPager container;
    int childLayoutResId;

    @SuppressWarnings("unused")
    private ContinuousPagerAdapter() {
    }

    /**
     * @param childLayoutResId Layout resource ID of the children to be inflated
     */
    public ContinuousPagerAdapter(int childLayoutResId, OnPageInstantiatedListener pListener) {
        this.childLayoutResId = childLayoutResId;
        this.pListener = pListener;
    }

    @Override
    public int getCount() {
        return 3;
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {
        return view == object;
    }

    @Override
    public Object instantiateItem(View container, int position) {
        this.container = (ViewPager) container;

        // inflate a new child view
        LayoutInflater li = (LayoutInflater) container.getContext().getSystemService(
                Context.LAYOUT_INFLATER_SERVICE);
        View childView = li.inflate(childLayoutResId, null, false);

        // add it to the view pager and return
        int count = this.container.getChildCount();
        int actualPos = count > position ? position : count;
        this.container.addView(childView, actualPos);
        pListener.onPageInstantiated(actualPos); // sometimes use 0 instead of actualPos, with different but still inconsistent results
        return childView;
    }

    @Override
    public void destroyItem(View container, int position, Object object) {
        ((ViewPager) container).removeViewAt(position);
    }

    public static interface OnPageInstantiatedListener {
        public void onPageInstantiated(int position);
    }

    /**
     * Needed to ensure all the items are instantiated
     */
    @Override
    public int getItemPosition(Object object) {
        return POSITION_NONE;
    }

    @Override
    public Parcelable saveState() {
        return null;
    }

    @Override
    public void restoreState(Parcelable state, ClassLoader loader) {
    }

    @Override
    public void finishUpdate(View container) {
    }
}

我不明白为什么页面在旋转后按 1->0->2 的顺序实例化。我也不保存状态。对此的任何见解都会有所帮助。


在仔细调试应用程序并查看之后ViewPager源码,我发现问题了。

应用程序第一次启动时mViewPager.setAdapter(mPagerAdapter)被调用后,页面会立即启动,应用程序将按其应有的方式工作。但是,当手机旋转时,调用setAdapter()推迟实例化页面,因为getWindowToken()由于窗口尚未准备好,因此返回 null。实例化被延迟,直到after onResume()在某个循环中被调用。

Calling setCurrentItem(1, false)使第一个页面成为主页面,因此它在其他页面之前被实例化,导致首先出现奇怪的 1->0->2 实例化。

解决方案? Use a Handler运行setCurrentItem()并在其他页面实例化后加载数据:

new Handler().post(new Runnable() {
    @Override
    public void run() {
        mViewPager.setCurrentItem(1, false);
        cleanupAndShowData();
    }
});

虽然我通常想避免使用Handlers,这似乎是迄今为止我找到的唯一选项,因为页面本身是在循环器中添加的。

EDIT: 即使上面也有一些问题。我最后打电话了mViewPager.setCurrentItem(1, false)仅在所有页面都已实例化之后(在onPageInstantiated()).

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

ViewPager 在屏幕旋转后重新实例化无序的项目 的相关文章

随机推荐