Как фрагмент загрузки в 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)
{
}
});