私のアプリには SlidingMenu があり、メニューの各項目は新しい SherlockFragment です。ほとんどは単純な 1 ペインのフラグメントですが、2 ペインにする必要があるフラグメントで問題が発生しています。ActionBar Tabs を備えた ViewPager を使用して実装することにしました。
ViewPager には、CustomerFragment (顧客のリスト) と CartFragment (カート項目のグリッドで、選択した顧客の名前を表示する TextView が上部にある) の 2 つのページがあります。ユーザーが人の名前をクリックすると、ViewPager がページをめくって名前を表示します。すべてが親フラグメントに含まれているため、ViewPager を実装する最善の方法は、ChildFragmentManager を使用することであると判断しました。
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
mViewPager = new ViewPager(activity);
mViewPager.setId(1);
v = mViewPager;
fm = getChildFragmentManager();
custFrag = CashRegisterCustomerFragment.newInstance();
cartFrag = CashRegisterCartFragment.newInstance();
fm.beginTransaction().add(mViewPager.getId(), custFrag).commitAllowingStateLoss();
fm.beginTransaction().add(mViewPager.getId(), cartFrag).commitAllowingStateLoss();
mActionBar = activity.getSupportActionBar();
mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
mTabsAdapter = new CashRegisterTabsAdapter(activity, mViewPager);
mTabsAdapter.addTab(mActionBar.newTab().setText(activity.getResources().getString(R.string.customer_list)),
custFrag);
mTabsAdapter.addTab(mActionBar.newTab().setText(activity.getResources().getString(R.string.items)),
cartFrag);
if (savedInstanceState != null) {
mActionBar.setSelectedNavigationItem(savedInstanceState.getInt("tab", 0));
} else {
mActionBar.setSelectedNavigationItem(0);
}
return v;
}
CashRegisterTabsAdapter の重要な部分:
public CashRegisterTabsAdapter(MyActivity act, ViewPager pager) {
super(act.getSupportFragmentManager());
activity = act;
mActionBar = activity.getSupportActionBar();
mActionBar.setTitle(R.string.cash_register);
mViewPager = pager;
mViewPager.setAdapter(this);
mViewPager.setOnPageChangeListener(this);
}
public void addTab(ActionBar.Tab tab, BaseFragment frag) {
tab.setTabListener(this);
mTabs.add(frag);
mActionBar.addTab(tab);
notifyDataSetChanged();
}
@Override
public int getCount() {
return mTabs.size();
}
@Override
public Fragment getItem(int position) {
return mTabs.get(position);
}
@Override
public void onPageSelected(int position) {
mActionBar.setSelectedNavigationItem(position);
if (position == 1)
activity.getSlidingMenu().setSlidingEnabled(false);
else
activity.getSlidingMenu().setSlidingEnabled(true);
}
@Override
public void onTabSelected(Tab tab, FragmentTransaction ft) {
int pos = tab.getPosition();
if (mViewPager != null) {
mViewPager.setCurrentItem(pos);
}
}
すべてがうまく機能します。ただし、 SlidingMenu を介してフラグメントを切り替えてからホームボタンを押して、アプリをバックグラウンドに送信すると、次のようになります。
07-02 18:04:31.170: E/AndroidRuntime(32619): FATAL EXCEPTION: main
07-02 18:04:31.170: E/AndroidRuntime(32619): java.lang.IllegalStateException: Failure saving state: active CashRegisterCustomerFragment{41a8f1a0} has cleared index: -1
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.support.v4.app.FragmentManagerImpl.saveAllState(FragmentManager.java:1700)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.support.v4.app.FragmentActivity.onSaveInstanceState(FragmentActivity.java:527)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.actionbarsherlock.app.SherlockFragmentActivity.onSaveInstanceState(SherlockFragmentActivity.java:127)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.jeremyfeinstein.slidingmenu.lib.app.SlidingFragmentActivity.onSaveInstanceState(SlidingFragmentActivity.java:50)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.mycom.myapp.MyActivity.onSaveInstanceState(MyActivity.java:128)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.mycom.myapp.MainFragmentContainer.onSaveInstanceState(MainFragmentContainer.java:124)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.Activity.performSaveInstanceState(Activity.java:1137)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.Instrumentation.callActivityOnSaveInstanceState(Instrumentation.java:1215)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.ActivityThread.performStopActivityInner(ActivityThread.java:3077)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.ActivityThread.handleStopActivity(ActivityThread.java:3136)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.ActivityThread.access$900(ActivityThread.java:142)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1235)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.os.Handler.dispatchMessage(Handler.java:99)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.os.Looper.loop(Looper.java:137)
07-02 18:04:31.170: E/AndroidRuntime(32619): at android.app.ActivityThread.main(ActivityThread.java:4931)
07-02 18:04:31.170: E/AndroidRuntime(32619): at java.lang.reflect.Method.invokeNative(Native Method)
07-02 18:04:31.170: E/AndroidRuntime(32619): at java.lang.reflect.Method.invoke(Method.java:511)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:791)
07-02 18:04:31.170: E/AndroidRuntime(32619): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:558)
07-02 18:04:31.170: E/AndroidRuntime(32619): at dalvik.system.NativeStart.main(Native Method)
奇妙なことに、CashRegister 親フラグメントのホーム ボタンを押してもクラッシュせず、最初に別の親フラグメントに移動したときにのみこの例外がスローされます。
このサイトでは、許可されていないときに FragmentManager がトランザクションを実行しようとしていると書かれていますが、それがどこにあるのかわかりません。onCreateView() で fm.beginTransaction.add() 呼び出しを削除すると、このエラーは発生しませんが、インターフェイス呼び出しで参照する親フラグメントがないため、フラグメント間で通信できません。