TL; DR:如果与数据绑定一起使用的布局具有EditText
,并且存在android:text
的绑定表达式,则绑定表达式将覆盖保存的实例状态值...即使我们不明确触发绑定评估。用户在配置更改消失之前键入的内容。我们如何解决这个问题,以便在配置更改时使用保存的实例状态值?
我们有一个愚蠢的Model
:
public class Model {
public String getTitle() {
return("Title");
}
}
我们有一个引用Model
:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="model"
type="com.commonsware.databindingstate.Model" />
</data>
<android.support.constraint.ConstraintLayout xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.commonsware.databindingstate.MainActivity">
<EditText android:id="@+id/title"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:inputType="text"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
</layout>
请注意,此布局没有绑定表达式;我们稍后会谈到这一点。
布局用于动态片段:
public class FormFragment extends Fragment {
@Nullable
@Override
public View onCreateView(LayoutInflater inflater,
@Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
return(MainBinding.inflate(inflater, container, false).getRoot());
}
}
请注意,我们并未在任何地方调用setModel()
来将Model
推送到绑定中。 MainBinding
(对于上面显示的main.xml
布局)仅用于夸大布局。
此代码(使用适当的FragmentActivity
设置FormFragment
)正确使用已保存的实例状态。如果用户在EditText
中输入内容,然后旋转屏幕,则新重新创建的EditText
会显示输入的文字。
现在,让我们更改布局以添加android:text
的绑定表达式:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="model"
type="com.commonsware.databindingstate.Model" />
</data>
<android.support.constraint.ConstraintLayout xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.commonsware.databindingstate.MainActivity">
<EditText android:id="@+id/title"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:inputType="text"
android:text="@{model.title}"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
</layout>
现在,如果用户在EditText
中输入内容并旋转屏幕,则新重新创建的EditText
为空。绑定表达式将覆盖从已保存的实例状态恢复的框架。
尽管事实上我没有在绑定上调用setModel()
,但这仍然存在。我当然可以看到,如果我在绑定上调用setModel()
,那么用模型中的数据替换EditText
内容。但我不这样做。
我可以在官方设备(Google Pixel,Android 8.0)和生态系统设备(Samsung Galaxy S8,Android 7.1)上重现此行为。
这可以通过自己保存状态并在某个时候恢复它来“手动”解决。例如,多个注释建议双向绑定,但这与其他设计目标(例如,不可变模型对象)相反。这似乎是数据绑定的一个相当基本的限制,所以我希望有一些我错过的东西,我可以配置为自动使用保存的实例状态。
答案 0 :(得分:2)
我认为ianhanniballake提到了一个相关的答案,但也许还有更多。以下是我对该参考如何应用于这些情况的解释。
使用您提供的XML,以下代码将从保存的实例状态中交替还原并从模型中还原。当恢复保存的实例状态时,可能是没有实例化的模型要从中恢复。那是mCount
是偶数的时候。如果存在模型,则基本上忽略保存的实例状态并且绑定接管。这里有一个比我们想要的更多的逻辑,但它不是明确地保存和恢复。
mCount
只是一种手段。将使用标志或是否存在模型的其他指示。
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding binding;
private int mCount;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
mCount = (savedInstanceState == null) ? 0 : savedInstanceState.getInt("mCount", 0);
if (mCount % 2 == 1) {
// 1st, 3rd, 5th, etc. rotations. Explicitly execute the bindings and let the framework
// restore from the saved instance state.
binding.executePendingBindings();
} else {
// First creation and 2nd, 4th, etc. rotations. Set up our model and let the
// framework restore from the saved instance state then overwrite with the bindings.
// (Or maybe it just ignores the saved instance state and restores the bindnings.)
Model model = new Model();
binding.setModel(model);
}
mCount++;
}
@Override
public void onSaveInstanceState(Bundle bundle) {
super.onSaveInstanceState(bundle);
bundle.putInt("mCount", mCount);
}
}