gpt4 book ai didi

Android 复合自定义 View 恢复保存状态

转载 作者:太空宇宙 更新时间:2023-11-03 13:47:45 24 4
gpt4 key购买 nike

我创建了复合自定义 View ,其中包含名为 LabledEditTextTextViewEditText,因为我将在一个 fragment 中有很多 EditText 字段。

我创建了一个包含以下内容的 XML 文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/linearLayoutLabeledEditText"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/label_textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/label"
android:textAppearance="?android:attr/textAppearanceMedium" />

<EditText
android:id="@+id/value_editText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:freezesText="true"
android:saveEnabled="true"/>
</LinearLayout>

在View类中如下

public class LabeledEditText extends LinearLayout {
private EditText editText;
private TextView label;

public LabeledEditText(Context context, AttributeSet attrs) {
super(context, attrs);
inflate(context,R.layout.labeled_edit_text, this);
label = (TextView) this.findViewById(R.id.label_textView);
label.setText("some label");
editText = (EditText) this.findViewById(R.id.value_editText);
}

protected void onRestoreInstanceState(Parcelable state) {
String id= this.getId()+" ";
if (state instanceof Bundle) // implicit null check
{
Bundle bundle = (Bundle) state;
state = bundle.getParcelable(id+"super");
super.onRestoreInstanceState(state);
editText.setText(bundle.getString(id+"editText"));
}
}

protected Parcelable onSaveInstanceState() {
String id= this.getId()+" ";
Bundle bundle = new Bundle();
bundle.putParcelable(id+"super",super.onSaveInstanceState());
bundle.putString(id+"editText",editText.getText().toString());
return bundle;
}
}

然后我在代表 3 个步骤的 3 个 fragment 中使用它。当我在第一个步骤/fragment 中插入值时

1st time after visiting the first step

然后切换到其他 fragment 并再次返回到第 1 步/fragment 我发现以下内容

2nd time visiting the first step

是什么导致了这个问题?

我已经调试了至少 5 天,请记住,在 fragment 布局中使用时,每个自定义 View 都有不同的 ID。

我也尝试在保存状态时将自定义 View 的 id 添加为键的一部分 this.getId()+"editText" 仍然是同样的问题。

编辑genrateViewId用于 API < 17

修改后的代码

import java.util.concurrent.atomic.AtomicInteger;

public class LabeledEditText extends LinearLayout {
private EditText editText;
private TextView label;

public LabeledEditText(Context context, AttributeSet attrs) {
super(context, attrs);
inflate(context,R.layout.labeled_edit_text, this);
label = (TextView) this.findViewById(R.id.label_textView);
editText = (EditText) this.findViewById(R.id.value_editText);
editText.setId(generateViewId());
applyAttr(context,attrs);
}

@Override
protected Parcelable onSaveInstanceState() {
Bundle bundle = new Bundle();
//adding the id of the parent view as part of the key so that
//editText state won't get overwritten by other editText
//holding the same id
bundle.putParcelable("super",super.onSaveInstanceState());
bundle.putString("editText",editText.getText().toString());
return bundle;
}

@Override
protected void onRestoreInstanceState(Parcelable state) {
if (state instanceof Bundle) // implicit null check
{
Bundle bundle = (Bundle) state;
state = bundle.getParcelable("super");
super.onRestoreInstanceState(state);
editText.setText(bundle.getString("editText"));
}
}

private static final AtomicInteger sNextGeneratedId = new AtomicInteger(1);

public static int generateViewId() {
for (;;) {
final int result = sNextGeneratedId.get();
// aapt-generated IDs have the high byte nonzero; clamp to the range under that.
int newValue = result + 1;
if (newValue > 0x00FFFFFF) newValue = 1; // Roll over to 1, not 0.
if (sNextGeneratedId.compareAndSet(result, newValue)) {
return result;
}
}
}
}

最佳答案

您的问题是您有多个 ViewGroups (LinearLayout),其中的子项具有相同的 ID。因此在保存状态时,所有的都保存在同一个状态,最后一个,覆盖所有。
要解决这个问题,您必须在膨胀时为每个 View 提供一个唯一的 I。在 v17 及更高版本中,您可以使用 View.generateViewId();,在旧版本中,您必须在 ids 文件中手动创建静态 ID。
您的代码应如下所示;

public LabeledEditText(Context context, AttributeSet attrs) { 
super(context, attrs); inflate(context,R.layout.labeled_edit_text, this);
label = (TextView) this.findViewById(R.id.label_textView); label.setText("some label");
editText = (EditText) this.findViewById(R.id.value_editText);
label.setId(View.generateViewId());
editText.setId(View.generateViewId());
}

在任何情况下,使用静态 ID 可能会更好,因为以后引用它们会更容易。您甚至可能不再需要覆盖 onSave 和 onRestore,尤其是当您使用静态 ID 时。

关于Android 复合自定义 View 恢复保存状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39191312/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com