奇怪的onPause(),片段中的onResume()行为

时间:2013-09-12 11:26:03

标签: android android-fragments android-tabs

应用程序有一个主要活动(MainActivity.java),带有三个选项卡(片段)。我可以使用向左滑动(riht)或单击特定选项卡在它们之间导航。

启动应用程序后,会显示第一个片段。

如果我从第一个片段转到第二个片段然后再回到第一个片段,则没有任何反应(第一个片段的onResume()未被调用),因此它不会刷新它的内容。

如果我从第一个片段转到第3个片段,然后直接返回到第一个片段,则创建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;
}
}


}

2 个答案:

答案 0 :(得分:2)

  

片段中的onPause(),onResume()行为()

这不是奇怪的行为,而是ActionSherlock的原生行为。这种行为用于缓存 - &gt;对于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()方法。此方法将用于从Fragment调用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()),您都可以进行片段更新。每次滚动或单击选项卡时,这种方式更有效地破坏和重新创建片段。但我怎么说这不仅是解决方案还有更多可能的解决方案。


注意: getItemPosition()可以返回两个值:POSITION_NONEUNCHANGED。两者之间的区别在于,第一个表示Fragment将始终被销毁并重新创建,效率不高,第二个表示Fragment不会被更改(位于正确的位置)。

有关详细说明look here

答案 1 :(得分:0)

这是因为ViewPager不会隐藏您切换的所有片段。 您可以通过setOffscreenPageLimit

控制此行为