Как фрагмент загрузки в ViewPager только тогда, когда выбранный

im, используя 3 фрагмента внутри Viewpager, проблема его загрузки больших данных в asytask и загрузчики, в устройствах, таких как HTC, хорошо работает, однако в устройствах с низким уровнем занимает много времени. Это связано с тем, что, главным образом, когда я реализую pagerAdapter, я помещаю фрагменты внутри arrayList, это заставляет фрагменты устанавливаться, когда основная активность загружается. То, что мне нужно, просто "загрузить" первый фрагмент (основной), и когда пользователь "swype" загрузит другой фрагмент. его каким-либо образом добиться этого? это моя страницаAdapater

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);

    }

} 

Ответ 1

Вышеупомянутый метод от Sun не работал у меня (может быть, он и для вас), но я думал, что также поделюсь своим редактированием его метода. Очень хороший метод кстати 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;
        }
    }
}

Ответ 2

Я собираюсь добавить свое решение здесь, так как столкнулся с аналогичной проблемой. Моя асинхронная задача не загружала огромные объемы данных, но предотвращала ненужные сетевые вызовы. Вот что я добавил в своем фрагменте:

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;
        }
    }
}

С вышеуказанным кодом ваш Фрагмент будет загружен, но ваша задача async не будет выполняться до тех пор, пока пользователь не начнет прокручиваться до Фрагмента в первый раз. После отображения ваша задача async будет выполняться в первый раз автоматически. Затем вы можете предоставить способ загрузить больше данных с помощью кнопки или потянуть для обновления. Вышеупомянутый фрагмент был в моем ViewPager и, казалось, работал нормально.

Ответ 3

Используйте fragStatePageAdapter, если у вас много страниц, и вы хотите их уничтожить, когда их не видно. Он реализовал setMenuVisibility (boolean menuVisible), когда фрагмент становится видимым, поэтому используйте это.

Ответ 4

Я могу опоздать на вечеринку, но здесь мое решение и работает, как ожидалось. Во всех ваших дочерних фрагментах создается переменная 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();
}

затем установите свой пейджер-адаптер и приложите к нему вкладку:

//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)
    {
    }
});