如何在旋转设备时保存在ViewModel中观察到的UI状态?

huangapple 未分类评论50阅读模式
英文:

How to save UI state observed in ViewModel when rotating device?

问题

我有一个显示地点详情的用户界面(UI),其中包括一些文本编辑框(EditTexts)和一个来自于 这个 ExpandableRecyclerView 库 的适配器(adapter)。

onCreate 中,当我的 ViewModel 观察到地点的变化时,我会更新 UI。我会在 onSaveInstanceState 中保存适配器,然后在 onRestoreInstanceState 中进行恢复。

然而,一旦我旋转设备,旧的 UI 状态就会显示出来。例如,我在文本编辑框中输入一些文本并展开适配器,但是在我旋转设备后,旧的文本和折叠的适配器会出现。我认为这是因为在 onChanged 被调用之前适配器为 null,因此在 onRestoreInstanceState 中跳过了我的适配器方法,然后在 onChanged 中 EditTexts 被重新更新为旧文本。

我该如何保存 UI 的状态?

完整的代码可以在这里找到。

onCreate:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_detail);

    viewModel = new ViewModelProvider(this).get(PlaceViewModel.class);

    Intent intent = getIntent();
    if (intent.hasExtra(EXTRA_PLACE_ID)) {
        String id = intent.getStringExtra(EXTRA_PLACE_ID);

        viewModel.getPlaceById(id).observe(this, new Observer<PlaceModel>() {
            @Override
            public void onChanged(PlaceModel placeModel) {
                // 如果地点未被删除,placeModel 将不为 null
                if (placeModel != null) {
                    // 在这里更新 UI
                }
            }
        });
    }
}

onSaveInstanceState 和 onRestoreInstanceState:

/* 要保存适配器的展开和折叠状态,
必须显式地调用适配器的 onSaveInstanceState() 和 onRestoreInstanceState() 方法 */
@Override
protected void onSaveInstanceState(@NonNull Bundle outState) {
    super.onSaveInstanceState(outState);
    if (adapter != null) {
        adapter.onSaveInstanceState(outState);
    }
}

@Override
protected void onRestoreInstanceState(@NonNull Bundle savedInstanceState) {
    super.onRestoreInstanceState(savedInstanceState);
    if (adapter != null) {
        adapter.onRestoreInstanceState(savedInstanceState);
    }
}
英文:

I have UI displaying details of a place, including some EditTexts and an adapter from this ExpandableRecyclerView library.

The UI is updated in onCreate when my ViewModel observes a change to the place. I save the adapter in onSaveInstanceState and restore it in onRestoreInstanceState.

However once I rotate my device, the old UI state is displayed. For example, I enter some text to the EditText and expand my adapter, but after I rotate my device the old text and collapsed adapter appears. I think this is because adapter is null until onChanged is called, so my adapter method in onRestoreInstanceState is skipped, and EditTexts are re-updated to old text in onChanged.

How do I save the state of my UI?

Full code can be found here.

onCreate:

protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_detail);

        viewModel = new ViewModelProvider(this).get(PlaceViewModel.class);

        Intent intent = getIntent();
        if (intent.hasExtra(EXTRA_PLACE_ID)) {
            String id = intent.getStringExtra(EXTRA_PLACE_ID);


            viewModel.getPlaceById(id).observe(this, new Observer&lt;PlaceModel&gt;() {
                @Override
                public void onChanged(PlaceModel placeModel) {
                    // placeModel will be null if place is deleted
                    if (placeModel != null) {
                        // update UI here
                    }
                }
            });
        }
}

onSaveInstanceState and onRestoreInstanceState:

/* To save the expand and collapse state of the adapter,
you have to explicitly call through to the adapter&#39;s
onSaveInstanceState() and onRestoreInstanceState() in the calling Activity */
@Override
protected void onSaveInstanceState(@NonNull Bundle outState) {
    super.onSaveInstanceState(outState);
    if (adapter != null) {
        adapter.onSaveInstanceState(outState);
    }
}

@Override
protected void onRestoreInstanceState(@NonNull Bundle savedInstanceState) {
    super.onRestoreInstanceState(savedInstanceState);
    if (adapter != null) {
        adapter.onRestoreInstanceState(savedInstanceState);
    }
}

答案1

得分: 0

一个解决方法就是检查 savedInstanceState 是否为 null,如果为 null,则不设置 EditText 的文本并恢复适配器状态。

但是我没有仔细阅读 ViewModel 文档: "ViewModel 类旨在以考虑生命周期的方式存储和管理与 UI 相关的数据。ViewModel 类允许数据在配置更改(如屏幕旋转)后继续存在。"

我之前在问为什么 ViewModel 在旋转后会保存我的旧 UI 相关数据...尽管这实际上正是它应该做的...这是我的错误。

英文:

One solution is just check if savedInstanceState is null, if it is don't set text of EditTexts and restore adapter state.

But I didn't carefully read the ViewModel documentation: "The ViewModel class is designed to store and manage UI-related data in a lifecycle conscious way. The ViewModel class allows data to survive configuration changes such as screen rotations."

I was asking why ViewModel is saving my old UI-related data after rotation...even though that's exactly what it's supposed to do...my mistake.

huangapple
  • 本文由 发表于 2020年7月24日 02:35:35
  • 转载请务必保留本文链接:https://java.coder-hub.com/63060954.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定