屏幕方向改变:当DialogFragment可见时出现非法状态异常

5
我希望让Android处理方向变化。不幸的是,每次打开对话框并发生方向变化时,它都会崩溃。我已经猜到了原因,但不知道该如何处理:
当应用程序首次启动时,返回堆栈看起来像这样:
02-07 15:25:36.517: I/System.out(14541): Added Fragments:
02-07 15:25:36.517: I/System.out(14541):   #0: PortraitModeFragment{41963330 #0 id=0x7f090001 ListViewFragment}

很好,现在我们打开一个对话框:

02-07 15:26:03.316: I/System.out(14541): Added Fragments:
02-07 15:26:03.316: I/System.out(14541):   #0: PortraitModeFragment{41963330 #0 id=0x7f090001 ListViewFragment}
02-07 15:26:03.326: I/System.out(14541):   #1: MyDialogFragment{4199f6a0 #1 MyDialogFragment}

现在旋转设备时:

02-07 15:26:37.502: E/AndroidRuntime(14541): FATAL EXCEPTION: main
02-07 15:26:37.502: E/AndroidRuntime(14541): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.testapp/com.example.testapp.MainActivity}: android.view.InflateException: Binary XML file line #11: Error inflating class fragment
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2180)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2230)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3692)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.access$700(ActivityThread.java:141)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1240)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.os.Handler.dispatchMessage(Handler.java:99)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.os.Looper.loop(Looper.java:137)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.main(ActivityThread.java:5039)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at java.lang.reflect.Method.invokeNative(Native Method)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at java.lang.reflect.Method.invoke(Method.java:511)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at dalvik.system.NativeStart.main(Native Method)
02-07 15:26:37.502: E/AndroidRuntime(14541): Caused by: android.view.InflateException: Binary XML file line #11: Error inflating class fragment
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:704)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.rInflate(LayoutInflater.java:746)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.inflate(LayoutInflater.java:489)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.inflate(LayoutInflater.java:396)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.inflate(LayoutInflater.java:352)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:270)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.Activity.setContentView(Activity.java:1881)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at com.example.testapp.MainActivity.onCreate(MainActivity.java:17)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.Activity.performCreate(Activity.java:5104)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1080)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2144)
02-07 15:26:37.502: E/AndroidRuntime(14541):    ... 12 more
02-07 15:26:37.502: E/AndroidRuntime(14541): Caused by: java.lang.IllegalStateException: Fragment com.example.testapp.fragments.LandscapeModeFragment did not create a view.
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.support.v4.app.FragmentActivity.onCreateView(FragmentActivity.java:303)
02-07 15:26:37.502: E/AndroidRuntime(14541):    at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:676)

有趣的事实是:如果我先旋转设备,然后打开对话框并更改方向,这种情况就不会发生,因为两个片段都预加载并在返回栈中和平共处。
02-07 15:34:00.015: I/System.out(15412):   #0: PortraitModeragment{419b99b0 #0 id=0x7f090001 ListViewFragment}
02-07 15:34:00.015: I/System.out(15412):   #1: LandscapeModeFragment{419b9b88 #1 id=0x7f090003 LandscapeModeFragment}
02-07 15:34:00.025: I/System.out(15412):   #2: MyDialogFragment{419d1da0 #2 MyDialogFragment}

我无法让用户在使用我的应用之前先旋转智能手机,那么该怎么办? ;)

复现步骤:

如何将对话框添加到后退栈中:

MyDialogFragment myTvShowDialog = new MyDialogFragment();
myTvShowDialog.show(getActivity().getSupportFragmentManager(),myTvShowDialog.TAG);

我所做的是为MainActivity创建了两个xml文件:

1. res/layout/main.xml(竖屏模式):

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:layout_height="match_parent">
     <TextView android:id="@+id/textView1" android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:text="layout-land/PORTRAIT" />
     <fragment class="com.example.testapp.fragments.PortraitModeFragment"
                        android:id="@+id/fragment" android:layout_weight="1"
                        android:layout_width="0px" android:layout_height="match_parent" android:tag="PortraitModeFragment">
     </fragment>
</LinearLayout>

2. res/layout-land/main.xml(横屏模式下的布局文件)

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:layout_height="match_parent" android:orientation="vertical">
    <TextView   android:id="@+id/textView1"   android:layout_width="wrap_content"     android:layout_height="wrap_content"     android:text="layout-land/LANDSCAPE" />
    <fragment class="com.example.testapp.fragments.LandscapeModeFragment"
            android:id="@+id/LandscapeModeFragment" android:layout_weight="1"
            android:layout_width="0px" android:layout_height="0dp" android:tag="LandscapeModeFragment">
     </fragment>
</LinearLayout>

MainActivity.java:

public class MainActivity extends FragmentActivity {
    private static final String TAG = "MainActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.v(TAG, "onCreate()");
        setContentView(R.layout.activity_main);
        getSupportFragmentManager().dump("", null,
                new PrintWriter(System.out, true), null);
    }
}

FragmentPortraitMode:

public class PortraitModeFragment extends Fragment {
    @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    View fragment = inflater.inflate(R.layout.portraitmodefragment_layout,
            null);
    Button findViewById = (Button) fragment.findViewById(R.id.button1);
    findViewById.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
            MyDialogFragment myDialog = new MyDialogFragment();
            myDialog.show(getActivity().getSupportFragmentManager(),
                    MyDialogFragment.TAG);
        }
    });
    TextView textView = (TextView) fragment.findViewById(R.id.textView1);
    textView.setText(TAG);
    return fragment;
}

片段横向模式:

public class LandscapeModeFragment extends Fragment {

    public static final String TAG = "LandscapeModeFragment";

    @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    View fragment = inflater.inflate(R.layout.landscapemode_fragment_layout,
            container, false);
    return fragment;
}
}

landscapemode_fragment_layout.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
            <TextView
            android:id="@+id/textView1"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="LANDSCAPEFRAGMENT" />
</LinearLayout>

portraitmode_fragment_layout_xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <Button   android:id="@+id/button1"    android:layout_width="wrap_content"    android:layout_height="wrap_content"     android:text="Button" />

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView" />

</LinearLayout>

'generic_dialog_layout.xml' 是什么,它为什么在 LandscapeModeFragment 中无法被填充? - D-Dᴙum
@Kerry添加了对话框的布局,命名方面有些不足,我承认。这只是一个通用的布局... - AndacAydin
我将使用的layout.xml的命名更改为landscape_fragment_layout.xml,现在在命名上应该更清晰。 - AndacAydin
1个回答

7

我找到了一个同样有效的解决方法。

首先:问题是什么?

在布局xml中定义的片段和通过FragmentManager添加的片段具有不同的生命周期,虽然可以使用FragmentManager干涉布局片段,但是意想不到的行为(异常)会导致应用程序崩溃。

正如Dianne Hackborn在Google Groups论坛中提到的:

定义XML中的片段主要用于保持。如果您要添加和删除,则应该始终以动态方式执行。

我可以查看这个未来版本,但请注意,无论我做什么,您都可以轻松地使自己陷入糟糕的情况 - 例如,如果您没有一个唯一的包含视图组用于该片段,那么当您在层次结构中删除/添加片段时,会产生不一致的行为。

因此,我们不允许混合使用FragmentManager-Fragments和LayoutXML-Fragments。这不幸地也涉及到通过Runtime添加的DialogFragments。发生了什么:Android-Framework在方向更改时将新创建的布局片段添加到后退栈中。 FragmentManager随后尝试在与新创建的Layout-Fragment相同的索引上添加DialogFragment。这就是混乱和IllegalStateException发生的时候。

那么,有什么解决方法?

大多数使用片段进行不同方向的示例都涉及定义FrameLayout作为容器,以将正确的Fragment放入其中。因此,我们将全程使用FragmentManager并禁止LayoutFragments从我们的设计中:

ActivityMain仅使用此简单xml作为其内容视图:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
            android:id="@+id/fragment_container_id"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent">
</FrameLayout>

并且在启动时,它将正确的片段注入容器中。
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.mainactivity_framelayout);
    createFragmentIfNotExits();
}
private void createFragmentIfNotExits() {
    int orientation = getResources().getConfiguration().orientation;
    Fragment newFragment;
    if (orientation == Configuration.ORIENTATION_PORTRAIT) {
        newFragment = new TvShowListPortraitModeFragment();
    } else {
        newFragment = new TvShowListLandscapeModeFragment();
    }
    FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
    ft.replace(R.id.fragment_container_id, newFragment);
    ft.commit();
}

缺点:不幸的是,无法通过这种方式保留片段的保存状态。或者至少我还需要弄清楚如何做到这一点。

以上是我的全部内容。我不会将此答案标记为“已回答”,因为我并没有真正使其工作,也许有人会找到更简单的解决方案并发布出来。然后当然我会将其标记为正确。


4
由于安卓开发团队的人们不费心去修复漏洞,而是推出带有新漏洞的新版本,我不认为会有比你描述的官方漏洞更好的解决方案。http://code.google.com/p/android/issues/detail?id=18529 - Lukas Hanacek
我发现在 http://code.google.com/p/android/issues/detail?id=18529#c8 中描述的替代解决方案非常有用。 - awy

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接