显示 DialogFragment 会引发“在 onSaveInstanceState 后无法执行此操作”错误

Ven*_*tis 2 android module appcelerator dialogfragment appcelerator-titanium

问题

嗨,我正在为 Titanium 创建一个 Android 和 iOS 模块,它有一个 sendLog 方法,该方法将一些任意 JSON 数据发送到服务器,如果它与某些预定义的过滤器匹配,则返回一个 URL。URL 应该在带有 webview 的模式对话框中打开。

我编写了本机 iOS 和 Android 库,并将它们包装为 Titanium 模块。在 iOS 上一切正常,但在 Android 上我无法打开对话框(请参阅下面的错误堆栈跟踪)。现在有一条日志消息总是为了测试目的而触发相同的网页。在 Android 上,它只是默默地失败。

测试用例

var mupets = require("be.iminds.mupets");
mupets.initialize("wappr", "http://tocker.iminds.be:3000/log/report.json", 1, 100, 3);
var esmLog = { 
    bar: "foo"
};
mupets.sendLog("es-test-01",JSON.stringify(esmLog));
Run Code Online (Sandbox Code Playgroud)

在这段代码之后(最多大约 10 秒后),模块应该显示一个带有以下网页的本机对话框:http : //tocker.iminds.be : 3000/es/sheets/test-01/index.html

相反,这是我一直得到的错误:

日志

java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
at android.app.FragmentManagerImpl.checkStateLoss(FragmentManager.java:1411)
at android.app.FragmentManagerImpl.enqueueAction(FragmentManager.java:1429)
at android.app.BackStackRecord.commitInternal(BackStackRecord.java:687)
at android.app.BackStackRecord.commit(BackStackRecord.java:663)
at android.app.DialogFragment.show(DialogFragment.java:256)
at be.iminds.mupets_client_android.logging.plugins.OutHttp.getEsm(OutHttp.java:122)
at be.iminds.mupets_client_android.logging.plugins.OutHttp$1.success(OutHttp.java:78)
at be.iminds.mupets_client_android.HttpClient$1$1.onResponse(HttpClient.java:76)
at okhttp3.RealCall$AsyncCall.execute(RealCall.java:133)
at okhttp3.internal.NamedRunnable.run(NamedRunnable.java:32)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1113)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
at java.lang.Thread.run(Thread.java:818)
Run Code Online (Sandbox Code Playgroud)

这是导致错误的Android代码:

Activity activity = (Activity) context;
                EsmDialogFragment esmDialogFragment = EsmDialogFragment.newInstance(new EsmDialogListener() {
                    @Override
                    public void submit(String type, JsonObject result) {
                        Mupets.sendLog(type, result);
                        esmShown = false;
                    }

                    @Override
                    public void onCancel(JsonObject cancelled) {
                        super.onCancel(cancelled);
                        Mupets.sendLog("ESM_cancelled", cancelled);
                        esmShown = false;
                    }
                }, url, true);
                FragmentTransaction transaction = activity.getFragmentManager().beginTransaction();
                Fragment prev = activity.getFragmentManager().findFragmentByTag(EsmDialogFragment.ESM_DIALOG_FRAGMENT);
                if (prev != null) {
                    transaction.remove(prev);
                }
                transaction.addToBackStack(null);
                Log.v(TAG, "Pre-show fragment");
                esmDialogFragment.show(transaction, EsmDialogFragment.ESM_DIALOG_FRAGMENT);
                Log.v(TAG, "Post-show fragment");
Run Code Online (Sandbox Code Playgroud)

Titanium 是否不允许使用 Fragments/或要求您在特定点调用 Dialog.show()?该错误涉及“...after onSaveInstanceState”,但我不知道如果我不创建活动,我将如何在 onSaveInstanceState 之前调用它,以及为什么当我在原生 Android 应用程序中使用它时代码有效。

这是一个 Titanium 示例项目,其模块在打开后应显示对话框:https : //www.dropbox.com/s/0v77xd5gllv6kb3/testModule.zip?dl=1

npa*_*ace 5

这不是一个小问题,因此没有快速简便的修复方法,您可以从答案中复制/粘贴。最重要的是,您将不得不重构一些代码。

您正在尝试显示 aDialogFragment以响应异步操作 - 如果该操作在 之后完成onSaveInstanceState,回调将尝试显示对话框并引发IllegalStateException.

保护自己免受此问题影响的方法是不要直接从回调中执行 UI 操作。取而代之的是,您需要推迟到开始或恢复ActivityFragment,以便您可以安全地显示对话框。

一个简单的方法是使用粘性事件,即从你的回调中发布一个粘性事件并在你的 UI 组件的onResume方法中订阅该类型的粘性事件。

如果您不想使用事件总线库,您可以改为从非 UI 组件调用异步方法,在回调中更新其内部状态,然后让 UI 组件在onResume. 如果使用这种方法,则需要小心管理全局状态。