web-dev-qa-db-ja.com

選択されている場合にのみViewPagerでフラグメントをロードする方法

私はFragments内で3 Viewpagerを使用していますが、問題はAsynctaskとローダーにビッグデータをロードしていることです。 HTC oneのようなデバイスでは問題なく動作しますが、ローエンドのデバイスでは時間がかかります。これは主に、pagerAdapterを実装するときに、フラグメントをArrayList内に配置するためです。これにより、メインアクティビティが読み込まれたときにフラグメントが強制的にインスタンス化されます。私が必要なのは、最初のフラグメント(メイン)を単に "ロード"し、ユーザーSwypeが他のフラグメントをロードすることです。これを達成する方法はありますか?これは私のページです

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

    }

} 
14
Javier

上記の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;
        }
    }
}
23

同様の問題に直面したので、ここに解決策を追加します。私の非同期タスクは大量のデータをロードしていませんでしたが、不要なネットワーク呼び出しを防ぎます。これが私のフラグメントに追加したものです:

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

上記のコードを使用すると、フラグメントがロードされますが、非同期タスクは、ユーザーが実際にフラグメントに初めてスクロールするまで実行されません。表示されると、非同期タスクが初めて自動的に実行されます。次に、ボタンを使用してさらにデータをロードする方法、またはプルして更新する方法を提供できます。上記のフラグメントは私のViewPagerにあり、正常に動作しているようです。

11
Sun

一部のビューが完全に初期化されていないことが原因で発生する可能性のある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;
            });
        }
    }
}
1
pjwin

私はパーティーに遅れるかもしれませんが、これが私の解決策であり、期待通りに機能します。すべての子フラグメントで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)
    {
    }
});
0
SolidSnake

多数のページがあり、表示されていないときにそれらを破棄する場合は、fragmentStatePageAdapterを使用します。フラグメントが表示されたときにsetMenuVisibility(boolean menuVisible)を実装しているので、それを使用してください。

0