Я разрабатываю приложение для Android для планшетов, а не , используя библиотеку совместимости.
Существует только одна активность, и она использует ActionBar с 3 вкладками.
В TabListener я использую setContentView, чтобы загрузить макет, специфичный для этой вкладки, а затем добавить соответствующие фрагменты в их фреймы.
Это почти работает точно так, как я хочу, за исключением случаев, когда вы достаточно быстро переключаетесь между вкладками, приложение вылетает.
Я использую Samsung Galaxy Tab в качестве устройства отладки, и переключение вкладок происходит очень быстро. В нормальном темпе я могу переключаться между ними, и страницы загружаются мгновенно. Проблема в том, когда я переключаюсь между вкладками.
Сначала я получил
IllegalStateException: Fragment not added
как видно здесь:
http://code.google.com/p/android/issues/detail?id=17029
Следуя предложению использовать блоки try / catch в onTabUnselected, я сделал приложение немного более надежным, но это привело к проблеме:
IllegalArgumentException: No view found for id 0x... for fragment ...
Я не обнаружил в Интернете ни одного другого случая с кем-либо, имеющего такую же проблему, поэтому я обеспокоен тем, что могу делать что-то, что не поддерживается. Опять же , я пытаюсь использовать 3 разных макета в одном действии - когда вы нажимаете на вкладку, слушатель вызывает setContentView для изменения макета, а затем добавляет фрагменты. Он прекрасно работает, если вы не начнете агрессивно переключаться между вкладками.
Я получил эту идею от: http://developer.android.com/resources/samples/ApiDemos/src/com/example/android/apis/app/FragmentTabs.html
и вместо TabListener, хранящего ссылку на один фрагмент, у меня есть их массив. Кроме того, я не использую attach / detach, так как они были добавлены только в API 13.
Моя теория состоит в том, что либо setContentView не завершил создание представлений, и поэтому FragmentTransaction не может добавить их, или ИЛИ фрагменты добавляются для одной вкладки, когда выбрана другая вкладка и вызывается setContentView, уничтожая другой набор. мнений.
Я пытался что-то взломать, чтобы замедлить переключение вкладок, но ничего не получилось.
Вот код для моего TabListener:
private class BTabListener<T extends Fragment> implements ActionBar.TabListener{
private int mLayout;
private Fragment[] mFrags;
private TabData mTabData;
private Activity mAct;
private boolean mNoNewFrags;
public BTabListener(Activity act, int layout, TabData td, boolean frags){
mLayout = layout;
mTabData = td;
mAct = act;
mNoNewFrags = frags;
mFrags = new Fragment[mTabData.fragTags.length];
for(int i=0; i<mFrags.length; i++){
//on an orientation change, this will find the fragments that were recreated by the system
mFrags[i] = mAct.getFragmentManager().findFragmentByTag(mTabData.fragTags[i]);
}
}
@Override
public void onTabReselected(Tab tab, FragmentTransaction ft) {
}
@Override
public void onTabSelected(Tab tab, FragmentTransaction ft) {
//this gets called _after_ unselected
//note: unselected wont have been called after an orientation change!
//we also need to watch out because tab 0 always gets selected when adding the tabs
//set the view for this tab
mAct.setContentView(mLayout);
for(int i=0; i<mFrags.length; i++){
//this will be null when the tab is first selected
if(mFrags[i]==null ){
mFrags[i] = Fragment.instantiate(GUITablet.this, mTabData.classes[i].getName());
}
//if there was an orientation change when we were on this page, the fragment is already added
if(!mNoNewFrags || mDefaultTab!=tab.getPosition()){
ft.add(mTabData.containterIDs[i], mFrags[i], mTabData.fragTags[i]);
}
}
mNoNewFrags = false;
}
@Override
public void onTabUnselected(Tab tab, FragmentTransaction ft) {
// this gets called when another tab is selected, before it's onSelected method
for(Fragment f : mFrags){
try{ //extra safety measure
ft.remove(f);
}catch(Exception e){
e.printStackTrace();
System.out.println("unselect couldnt remove");
}
}
}
}
И, наконец, трассировка стека:
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): java.lang.IllegalArgumentException: No view found for id 0x7f0b0078 for fragment Fraggle{40ab2230 #2 id=0x7f0b0078 dummy2}
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:729)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:926)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.BackStackRecord.run(BackStackRecord.java:578)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1226)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.FragmentManagerImpl$1.run(FragmentManager.java:374)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.os.Handler.handleCallback(Handler.java:587)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.os.Handler.dispatchMessage(Handler.java:92)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.os.Looper.loop(Looper.java:132)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at android.app.ActivityThread.main(ActivityThread.java:4028)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at java.lang.reflect.Method.invokeNative(Native Method)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at java.lang.reflect.Method.invoke(Method.java:491)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:844)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
09-29 01:53:08.200: ERROR/AndroidRuntime(4611): at dalvik.system.NativeStart.main(Native Method)
СПАСИБО !!