8

FragmentStatePagerAdapterを使用してフラグメント(arrayListOfObjectsからのオブジェクトを表す)を取り込んでいるViewPagerがあります。すべてうまくいきます:

            mMyFragmentPagerAdapter = new fragmentAdapter(getSupportFragmentManager(),orientation
            ,rePopulatedfireInfoList);
        mPager = (ViewPager)findViewById(R.id.fireInfoFragment_container);
        initButton();
        setTab();
        mPager.setAdapter(mMyFragmentPagerAdapter); 

フラグメント アダプターは、FragmentStatePagerAdapter を拡張します。

プライマリ アクティビティから、ダイアログをテーマにしたアクティビティを起動します。ここで、ユーザーは新しいお気に入りの場所を追加して、プライマリ アクティビティによって渡されるオブジェクトの配列リストを変更する新しいオブジェクトを作成できます。これは、ダイアログ アクティビティを開始するためのコードです。すべて正常に動作します:

        Intent locationIntent = new Intent(afisController.this, locationActivity.class);
    locationIntent.putExtra("firesList", new fireInfoListWrapper(arrayListOfObjects));
    startActivityForResult(locationIntent,1);

フローティング アクティビティは、オブジェクトを arrayListOfObjects に追加します。

プライマリ アクティビティの onActivityResult で、受信した arraListOfObjects と送信したものを比較します。異なる場合は、viewPager の内容を完全に削除し、新しい arrayListOfObjects で再作成します。これは onActivityResults です:

     protected void onActivityResult(int requestCode, int resultCode,
            Intent data) {
 Toast.makeText(this, "Activity Results fired..." , 1500 ).show();
            if ((resultCode == 0) && (data != null)) { 
                Log.i("onActvityResult", "Inside resultCode check");   
                Bundle b = data.getExtras();
                if(b != null){
                    Log.i("onActvityResult", "not null");
                    returnedFireInfoList = (ArrayList<fireInfo>) data.getSerializableExtra("firesListResult"); 
                    Log.i("onActvityResult", "results Size: "+returnedFireInfoList.size());
                    if(returnedFireInfoList.size()>0){
                        Log.i("onActvityResult", "locationName: "+returnedFireInfoList.get(0).getLocationName());
                        //compare returnedFireInfoList and rePopulatedfireInfoList, if different; 
                        //add difference to rePopulatedfireInfoList and write back to file.
                        updateFireInfos(returnedFireInfoList, rePopulatedfireInfoList);
        if(returnedFireInfoList.size()!=rePopulatedfireInfoList.size()){
mMyFragmentPagerAdapter1 = new fragmentAdapter(getSupportFragmentManager(),orientation
                    ,returnedFireInfoList);
            mPager = (ViewPager)findViewById(R.id.fireInfoFragment_container); 
            Log.i("updateFireInfos", "fragmentsCount is"+mPager.getCurrentItem());
            fireInfoFragment fragment = 
                      (fireInfoFragment) getSupportFragmentManager().findFragmentById(R.id.fireInfoFragment_container);
//This is where the problem is, I don't want to remember what was already on the viewPager //called mPager before.
           //   mPager.removeAllViews();
            //mPager.setAdapter(null);
            mMyFragmentPagerAdapter1.notifyDataSetChanged();            
            mPager.setAdapter(mMyFragmentPagerAdapter1);               mMyFragmentPagerAdapter1.notifyDataSetChanged();                     
                    }                   
                }
            }        

これは fragmentStateAdapter コードです:

public class fragmentAdapter extends FragmentStatePagerAdapter {

private FragmentManager fragmentManager;
 private FragmentTransaction mCurTransaction = null;
 private ArrayList<Fragment.SavedState> mSavedState = new ArrayList<Fragment.SavedState>();
 private ArrayList<Fragment> mFragments = new ArrayList<Fragment>();
 private Fragment mCurrentPrimaryItem = null;



public void restoreState(Parcelable state, ClassLoader loader) {
    //Need to only delete info from marked fragments (theoned that are stored on orientationchange
    //Currently redoing the entire call; resulting in delay due to server call
    //if(isLastOrientationPortrait != isPortrait){
    if(state != null){
        Bundle bundle1 = (Bundle) state;
        bundle1.setClassLoader(loader);
        Iterable<String> keys = bundle1.keySet();
        Log.i("restoreState", "containsKey FragmentStatePagerAdapter: "+keys);
        //android.support.v4.app.FragmentManager fragmentManager= fragmentAdapter.this.fragmentManager;
        android.support.v4.app.FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); 
        //if (fragmentTransaction == null) {
        //  Log.i("restoreState", "fragmentTransactionTest");

            fragmentTransaction = fragmentManager.beginTransaction();
       // }
        for (String key : keys) {
            if (key.startsWith("f")) {
                Fragment f = fragmentManager.getFragment(bundle1,
                        key);
                fragmentTransaction.remove(f);
                fragmentTransaction.commit();
            }
            }
        }
    //}       
    }


@Override
public int getItemPosition(Object object) {
    // TODO Auto-generated method stub
    //return super.getItemPosition(object);
    return fragmentAdapter.POSITION_NONE;
}


public fragmentAdapter(android.support.v4.app.FragmentManager fragmentManager,String orientation,
         ArrayList<fireInfo> fireInfoList) {
     super(fragmentManager);
     this.orientation = orientation;
     this.fireInfoList = fireInfoList;
     this.numItems = fireInfoList.size();
     this.fragmentManager=fragmentManager;
 }
 ArrayList<fireInfo> fireInfoList;
 String orientation;
 int numItems;


@Override
 public int getCount() {
     Log.i("numItems", "is: "+fireInfoList.size());
     return numItems;
 }

@Override
public Fragment getItem(int arg0) {
    Log.i("fragmentAdapterIndex", "is: "+arg0);
    return fireInfoFragment.newInstance(orientation, fireInfoList.get(arg0));
}    

}

問題: しかし、startActivityFor の結果を起動する前に、新しい ArrayListOfObjects が古いものと一緒に追加されます。

viewPager に古いコンテンツを強制的に忘れさせるにはどうすればよいですか? 基本的に、fragmentStateAdapter を使用して、この newArrayListofObjects で viewPager アダプターをリセットしますか?

4

2 に答える 2

52

問題は、アダプターに使用する古いフラグメントがまだ残っているという事実にあると思いFragmentManagerます。この場合は、古いフラグメントをすべてフラグメント マネージャから削除するだけです。

したがって、基本的には、アダプターのコンストラクターで次のコードを実行するだけです。

public fragmentAdapter(FragmentManager fragmentManager, String orientation, ArrayList<fireInfo> list) {
    super(fragmentManager);
    if (fragmentManager.getFragments() != null) {
        fragmentManager.getFragments().clear();
    }
    //... your other code here
}

このコード行は不要です。

mMyFragmentPagerAdapter1.notifyDataSetChanged();     

編集:次を使用してフラグメントを削除する方が正しい場合がありますFragmentTransaction:

    List<Fragment> fragments = fragmentManager.getFragments();
    if (fragments != null) {
        FragmentTransaction ft = fragmentManager.beginTransaction();
        for (Fragment f : fragments) {
            //You can perform additional check to remove some (not all) fragments:
            if (f instanceof AddedByCurrentPagerAdapterFragment) { 
                ft.remove(f);
            }
        }
        ft.commitAllowingStateLoss();
    }

FragmentTransactionこれは(非同期で) 実行されるまでに時間がかかります。

于 2013-11-19T10:04:32.107 に答える
0

FragmentStatePagerAdapter を再割り当てするだけで済みました。

これは onActivityResult に見られるように配置されていますが、正しい動作を隠していたのは、viewPager ページ インジケーターでした。たとえば、viewPager に 2 つのページがあり、arrayListOfObjects に 1 つのオブジェクトを追加する子アクティビティを呼び出した場合に; viewPager ページ インジケーターは、現在 2 プラス 3 ページ (5) であることを示します。

フローティング アクティビティによって返されたばかりのこの新しい arrayListOfObjects によって決定されるようにするには、onActivityResult の viewPage インジケーターをリセットする必要がありました。

于 2013-05-16T15:11:46.763 に答える