如何在ViewPager中加载片段时才选择它

时间:2013-10-21 03:31:54

标签: java android android-fragments android-viewpager

我在Fragments内使用3 Viewpager,问题是我在Asynctask和加载器中加载大数据。在像HTC one这样的设备上,它工作正常,但是,在低端设备上,它需要花费很多时间。这主要是因为当我实现pagerAdapter时,我将片段放在ArrayList内,这会强制片段在加载主活动时实例化。我需要的是它只是“加载”第一个片段(main),当用户Swype加载另一个片段时。它的任何方式实现这一目标?这是我的pageAdapater

public class PagerAdapter extends FragmentPagerAdapter  {

    private final ArrayList<Fragment> mFragments = new ArrayList<Fragment>();
   // private final ArrayList<String> titulos = new ArrayList<String>();

   // private int NUM_PAGES =0;
    public PagerAdapter(FragmentManager manager,int num_pages) {
        super(manager);
     //   this.NUM_PAGES = num_pages;
    }


    public void addFragment(Fragment fragment,String title) {
        mFragments.add(fragment);
            notifyDataSetChanged();
    }

    @Override
    public int getCount() {
        //return NUM_PAGES;
       return mFragments.size();
    }

    @Override
    public Fragment getItem(int position) {
        return mFragments.get(position);

    }

} 

5 个答案:

答案 0 :(得分:17)

来自Sun的上述方法对我不起作用(也许它适合你),但我想我也会分享我对他的方法的编辑。太阳方式非常好的方法!

private boolean _hasLoadedOnce= false; // your boolean field

@Override
public void setUserVisibleHint(boolean isFragmentVisible_) {
    super.setUserVisibleHint(true);


    if (this.isVisible()) {
        // we check that the fragment is becoming visible
        if (isFragmentVisible_ && !_hasLoadedOnce) {
            new NetCheck().execute();
            _hasLoadedOnce = true;
        }
    }
}

答案 1 :(得分:10)

我要在这里添加我的解决方案,因为我遇到了类似的问题。我的异步任务没有加载大量数据,但它可以防止不必要的网络调用。这是我在片段中添加的内容:

private boolean _hasLoadedOnce= false; // your boolean field

@Override
public void setUserVisibleHint(boolean isFragmentVisible_) {
    super.setUserVisibleHint(isVisibleToUser);


    if (this.isVisible()) {
        // we check that the fragment is becoming visible
        if (!isFragmentVisible_ && !_hasLoadedOnce) {
            //run your async task here since the user has just focused on your fragment
            _hasLoadedOnce = true;
        }
    }
}

使用上面的代码,您的Fragment将被加载,但是您的异步​​任务将不会运行,直到用户第一次实际滚动到Fragment。显示后,您的异步任务将自动第一次运行。然后,您可以提供一种通过按钮加载更多数据或拉动刷新的方法。上面的片段在我的ViewPager中,似乎工作正常。

答案 2 :(得分:0)

如果您有很多页面并且想要在不可见时销毁它们,请使用fragmentStatePageAdapter。 当片段变得可见时,它已经实现了setMenuVisibility(布尔menuVisible),所以使用它。

答案 3 :(得分:0)

我可能会参加聚会,但这是我的解决方案,它按预期工作。在您的所有子片段中创建一个boolean变量:

private boolean loadFragmentExecuted = false;
子片段中的

创建一个名为loadFragment的通用方法,并将您在onCreateView中添加的所有逻辑移动到该方法:

public void loadFragment()
{
    if(!loadFragmentExecuted)
    {
        //Add your logic to manipulate the UI or load data etc...
        loadFragmentExecuted = true;
    }
}
在您的网页浏览逻辑中,

动态创建片段,如:

//add the fragment
String fragmentName = "com.something." + fragmentId;

//check if the class exists
try
{
    Class myFragmentClass = Class.forName(fragmentName);
    Fragment myFragment = (Fragment) myFragmentClass.newInstance();
    mFragments.add(myFragment);
}
catch (ClassNotFoundException e)
{
    e.printStackTrace();
}
catch (IllegalAccessException e)
{
    e.printStackTrace();
}
catch (InstantiationException e)
{
    e.printStackTrace();
}

然后设置你的寻呼机适配器并附上一个tablayout:

//set our pager adapter that contains different fragments
mPagerAdapter = new BasePagerAdapter(mFragmentManager, mFragments);

//link the adapter to the viewpager
mViewPager.setAdapter(mPagerAdapter);

//cache fragments
int limit = (mPagerAdapter.getCount() > 0 ? mPagerAdapter.getCount() : 1);
mViewPager.setOffscreenPageLimit(limit);

//add the page listner to the viewPager and link it to the tabLayout
mViewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));

//on tab selected select current viewpager item
mTabLayout.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener()
{
    @Override
    public void onTabSelected(TabLayout.Tab tab)
    {
        mViewPager.setCurrentItem(tab.getPosition());

        //get fragment for the selected tab
        Fragment f = mPagerAdapter.getItem(tab.getPosition());

        //load the content of the fragment
        try
        {
            Class c = f.getClass();
            Method loadFragment = c.getMethod("loadFragment");
            loadFragment.invoke(f);
        }
        catch (IllegalAccessException e){}
        catch (InvocationTargetException e){}
        catch (NoSuchMethodException e){}
    }

    @Override
    public void onTabUnselected(TabLayout.Tab tab)
    {
    }

    @Override
    public void onTabReselected(TabLayout.Tab tab)
    {
    }
});

答案 4 :(得分:0)

稍作修改的版本,以修复由于某些视图未完全初始化而导致的潜在NPE

private boolean _hasLoadedOnce= false; // your boolean field

@Override
public void setUserVisibleHint(boolean isFragmentVisible_) {
    super.setUserVisibleHint(isVisibleToUser);


    if (this.isVisible()) {
        // we check that the fragment is becoming visible
        if (!isFragmentVisible_ && !_hasLoadedOnce) {
            new Handler().post(() -> {
                makeAsyncRequest();//do your asyn stuffs 
                _hasLoadedOnce = true;
            });
        }
    }
}