1

アプリケーションには、3 つのタブ (フラグメント) を持つメイン アクティビティ (MainActivity.java) があります。左にスワイプ (右) するか、特定のタブをクリックして、それらの間を移動できます。

アプリケーションを起動すると、最初のフラグメントが表示されます。

1 番目のフラグメントから 2 番目のフラグメントに移動してから 1 番目のフラグメントに戻ると、何も起こらない (1 番目のフラグメントの onResume() は呼び出されない) ため、コンテンツは更新されません。

1 番目のフラグメントから 3 番目のフラグメントに移動し、1 番目のフラグメントに直接戻ると、fragment1 の onCreateView() が作成され、onResume() が呼び出されます。これは正しいです。

3 番目のフラグメントから 2 番目のフラグメントに移動すると、fragment1 の onCreateView() と onResume() が呼び出されますが、fragment2 の onCreateView は呼び出されません。

MainActivity のロジックが正しくないと思うので、誰かに見てもらい、何が間違っているのか教えてください。

MainActivity.java:

public class MainActivity extends FragmentActivity implements ActionBar.TabListener {

CollectionPagerAdapter mCollectionPagerAdapter;
public TTSocket socket;
DBHandler db;
public String logged_user;
private LogedinPerson person;
ViewPager mViewPager;

public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.main);

    Bundle extras = getIntent().getExtras();
    logged_user = extras.getString("logged_user");
    socket = TTSocket.getInstance();
    socket.currentRef = this;

    db = new DBHandler(this);
    person=db.getLogedInPerson();

    socket.dbHandler=db;
    socket.person=person;

    if(!socket.isInit){
        String typeInitStr = "{\"Type\":\"Init\", \"UserId\":\""+ person.getUserId() +"\"}";
        socket.Send(typeInitStr);
    }


    mCollectionPagerAdapter = new CollectionPagerAdapter(getSupportFragmentManager());

    // Set up action bar.
    final ActionBar actionBar = getActionBar();

    // Specify that we will be displaying tabs in the action bar.
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);

    // Set up the ViewPager, attaching the adapter and setting up a listener
    // for when the
    // user swipes between sections.
    mViewPager = (ViewPager) findViewById(R.id.pager);
    mViewPager.setAdapter(mCollectionPagerAdapter);
    mViewPager.setOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() {
            @Override
            public void onPageSelected(int position) {
            Log.d("TEST",position+"");
            // the corresponding tab.
            // We can also use ActionBar.Tab#select() to do this if
            // we have a reference to the Tab
            actionBar.setSelectedNavigationItem(position);
            }
        });

    // For each of the sections in the app, add a tab to the action bar.
    for (int i = 0; i < mCollectionPagerAdapter.getCount(); i++) {
        // Create a tab with text corresponding to the page title defined by
        // the adapter.
        // Also specify this Activity object, which implements the
        // TabListener interface, as the
        // listener for when this tab is selected.
        if(i == 0){
            actionBar.addTab(actionBar.newTab()
                    .setIcon(R.drawable.messages)
                    .setTabListener(this));

        }else if(i == 1){
            actionBar.addTab(actionBar.newTab()
                    .setIcon(R.drawable.contacts)
                    .setTabListener(this));
        }else{
            actionBar.addTab(actionBar.newTab()
                    .setIcon(R.drawable.history)
                    .setTabListener(this));
        }
    }
}

@Override
protected void onResume() {
    // TODO Auto-generated method stub
    super.onResume();

    socket.currentRef = this;
    socket.dbHandler=db;
    socket.person=person;

    //mCollectionPagerAdapter.notifyDataSetChanged();
}

public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
}

public void onTabSelected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
// When the given tab is selected, switch to the corresponding page in
// the ViewPager.
mViewPager.setCurrentItem(tab.getPosition());
}

public void onTabReselected(ActionBar.Tab tab, FragmentTransaction fragmentTransaction) {
}

/**
 * A {@link FragmentPagerAdapter} that returns a fragment corresponding to
 * one of the primary sections of the app.
 */
public class CollectionPagerAdapter extends FragmentPagerAdapter {

final int NUM_ITEMS = 3; // number of tabs

public CollectionPagerAdapter(FragmentManager fm) {
    super(fm);
}
/*
@Override
public Fragment getItem(int i) {
    Fragment fragment = new TabFragment();
    Bundle args = new Bundle();
    args.putInt(TabFragment.ARG_OBJECT, i);
    fragment.setArguments(args);
    return fragment;
}
*/
@Override  
public Fragment getItem(int position) {  

    Fragment fragment = new Fragment(); 
    Bundle args = new Bundle();
    args.putInt(TabFragment.ARG_OBJECT, position);
    switch (position) {  
        case 0:  
            Log.i("Fragment", "0");
            fragment =  new Tab1Fragment();
            fragment.setArguments(args);
            return fragment; 
        case 1:  
            Log.i("Fragment", "1");
            fragment =  new Tab2Fragment();  
            fragment.setArguments(args);
            return fragment; 
        case 2:  
            Log.i("Fragment", "2");
            fragment =  new Tab3Fragment();   
            fragment.setArguments(args);
            return fragment;  
        default:  
            break;  
    }

    return fragment;
}

@Override
public int getCount() {
    return NUM_ITEMS;
}

@Override
public CharSequence getPageTitle(int position) {
    String tabLabel = null;
    switch (position) {
    case 0:
    tabLabel = getString(R.string.label1);
    break;
    case 1:
    tabLabel = getString(R.string.label2);
    break;
    case 2:
    tabLabel = getString(R.string.label3);
    break;
    }

    return tabLabel;
}
}

/**
 * A fragment that launches other parts of the demo application.
 */
public static class TabFragment extends Fragment {

public static final String ARG_OBJECT = "object";

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
    Bundle savedInstanceState) {

    Bundle args = getArguments();
    int position = args.getInt(ARG_OBJECT);

    int tabLayout = 0;
    switch (position) {
    case 0:
        tabLayout = R.layout.tab1;
        break;
    case 1:
        tabLayout = R.layout.tab2;
        break;
    case 2:
        tabLayout = R.layout.tab3;
        break;
    }

    View rootView = inflater.inflate(tabLayout, container, false);

    return rootView;
}
}


}
4

2 に答える 2

2

フラグメント内の奇妙な onPause()、onResume() の動作()

おかしな動作ではなく、ActionSherlock 本来の動作です。RAMこの種の動作はキャッシュに使用されます ->断片がキャッシュされる理由は、これが低い古いデバイスの最適化です。

フラグメントのコンテンツを更新する必要がある場合は、レイアウトなどを置き換えようとしないでください。ページ間をスクロールするときにフラグメントを更新する場合は、FragmentPagerAdapter のメソッドを使用する必要があります。

@Override
public int getItemPosition(Object object) {
   // implementation
   return super.getItemPosition(object);
}

このメソッドは、呼び出し時に呼び出されます

notifyDataSetChanged();

FragmentPagerAdapter で。フラグメントを更新するための便利な方法です。方法は他にもありますが、ここでは私が行っている方法を紹介します。

たとえば Updateable と呼ばれるインターフェイスをフラグメントに実装させます。

interface Updateable {
   public void update();
}

public class MyFragment extends SherlockFragment implements Updateable {

   @Override
   public void update() {
      // perform Fragment updates
   }
}

この方法では、更新を実行します。メソッドに戻りgetItemPosition()ます。update()このメソッドは、フラグメントからメソッドを呼び出すために使用されます。つまり、次のようになります。

@Override
   public int getItemPosition(Object object) {
      Fragment f = (Fragment) object;
      // determine which fragment
      if (f instanceof MyFragment) {
         ((MyFragment) f).update(); // invokes update() method
      }

      return super.getItemPosition(object);
   }

これで、ページをスクロールするか、タブをタップするたびに ( も呼び出す必要がありますnotifyDataSetChanged())、Fragment を更新できます。この方法は、タブをスクロールまたはクリックするたびにフラグメントを破棄および再作成するよりも効率的です。しかし、これが唯一の解決策ではなく、さらに多くの解決策があると私が言ったことです。


注:getItemPosition()の 2 つの値を返すことができます。両方の違いは、1 つ目は Fragment が常に破棄されて再作成されることを示しており、これはあまり効率的ではなく、2 つ目は Fragment が変更されない (正しい場所にある) ことを示しています。POSITION_NONEUNCHANGED

詳細な説明については、こちらをご覧ください

于 2013-09-12T12:45:50.380 に答える