如何根据另一个 EditText 的输入更改一个 EditText 的值
How do I change value of one EditText based on the input in another
我目前有一个摄氏温度的 EditText 和一个华氏温度的 TextView 来显示答案。我使用了 TextChangedListener 但这不适用于 EditText。我无法使用 TextChangedListener 更改 EditText 中的值。有什么办法可以做到这一点。?我希望用户在任何 EditTexts 中输入一个值并在另一个中获得结果。这是我的 TextView 代码。我的主要活动:
public class MainActivity extends ActionBarActivity {
EditText C, F;
TextView Ct, Ft;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
C = (EditText) findViewById(R.id.celsius);
F = (EditText) findViewById(R.id.fahrenheit);
Ct = (TextView) findViewById(R.id.celsiusTextView);
Ft = (TextView) findViewById(R.id.fahrenheitTextView);
C.addTextChangedListener(new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
F.addTextChangedListener(new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (!s.toString().equals("")) {
f = Double.parseDouble(s.toString());
c = ((f - 32) * 5 / 9);
Ct.setText(String.valueOf(c));
} else {
Ct.setText("");
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
}
}
还有我的XML:
<RelativeLayout
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"
tools:context=".MainActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="200dp"
android:id="@+id/nameText"
android:background="@drawable/fab_linear_layout">
<TextView
android:text="@string/converter"
android:layout_gravity="bottom"
android:textSize="30sp"
android:textColor="#FFFFFF"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:padding="20dp"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_below="@+id/nameText"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:id="@+id/linearLayout1"
android:layout_centerVertical="true"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true">
<EditText
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:inputType="numberSigned|numberDecimal"
android:hint="@string/celsius"
android:id="@+id/celsius"/>
<TextView
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:hint="@string/fahrenheit"
android:id="@+id/fahrenheitTextView"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_below="@+id/linearLayout1"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:id="@+id/linearLayout2"
android:layout_centerVertical="true"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true">
<EditText
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:inputType="numberSigned|numberDecimal"
android:hint="@string/fahrenheit"
android:id="@+id/fahrenheit"/>
<TextView
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:hint="@string/celsius"
android:id="@+id/celsiusTextView"/>
</LinearLayout>
从代码可以看出,我想去掉多余的TextView,在EditText中得到答案。
您在错误的视图上设置了文本:
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
而不是 Ft
这是一个 TextView
你应该在 F
上调用 setText
这是 EditText。
看看你的 post 你正在进入一个无限循环,应用崩溃了。由于您在两个 EditText 中都有 onTextChanged 方法,因此当 C 更改文本时,更改 F 会更改 C,因为已更改.... 一次又一次。
做一个简单的测试,如果您从 C 或 F 中删除其中一个方法,您会发现更改正在起作用。
您需要做的是区分用户更改的文本和应用程序自动更改的文本。
我的建议如下
定义一个布尔值来进行微分:
EditText C, F;
TextView Ct, Ft;
private Boolean machine_changed_edittext = false;
不是在每个 onTextChanged 添加以下内容:
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
//We check that the change is human
if (! machine_changed_edittext) {
machine_changed_edittext = true;
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
F.setText(String.valueOf(f));
} else {
F.setText("");
}
//Now as it is finished we declare back the Boolean to the correct value
machine_changed_edittext = false;
}
}
我想我应该做对了,但这只是一个你把布尔值改变的游戏。如果我的想法行不通,请告诉我,我们开始聊天,我们可以调试它。
//未经测试的代码。
如果你只想使用两个EditText。如果您从 onTextChanged
中的其他 EditText 中删除文本观察器,它可能会起作用。
假设 Fahrenheit EditText 的文本观察器称为 fWatcher,Celcius EditText 的文本观察器称为 cWatcher。
想法是,当您在摄氏度编辑文本中修改文本时,删除华氏度编辑文本上的文本观察器,这样它就不会在您编辑文本时触发该监听器。
像这样定义 cWatcher :
TextWatcher cWatcher = new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
fEditText.removeTextChangedListener(fWatcher);
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
}
fEditText.setTextChangedListener(fWatcher);
}
@Override
public void afterTextChanged(Editable s) {
}
}
同样定义另一个TextWatcher。
实现此目的的另一种方法是检查 EditText 是否具有焦点(Kotlin 代码):
inputOne.addTextChangedListener { editableString->
//do nothing if this EditText is not being edited.
if(!inputOne.hasFocus())
return@addTextChangedListener
//do stuff
}
我目前有一个摄氏温度的 EditText 和一个华氏温度的 TextView 来显示答案。我使用了 TextChangedListener 但这不适用于 EditText。我无法使用 TextChangedListener 更改 EditText 中的值。有什么办法可以做到这一点。?我希望用户在任何 EditTexts 中输入一个值并在另一个中获得结果。这是我的 TextView 代码。我的主要活动:
public class MainActivity extends ActionBarActivity {
EditText C, F;
TextView Ct, Ft;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
C = (EditText) findViewById(R.id.celsius);
F = (EditText) findViewById(R.id.fahrenheit);
Ct = (TextView) findViewById(R.id.celsiusTextView);
Ft = (TextView) findViewById(R.id.fahrenheitTextView);
C.addTextChangedListener(new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
F.addTextChangedListener(new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
if (!s.toString().equals("")) {
f = Double.parseDouble(s.toString());
c = ((f - 32) * 5 / 9);
Ct.setText(String.valueOf(c));
} else {
Ct.setText("");
}
}
@Override
public void afterTextChanged(Editable s) {
}
});
}
}
还有我的XML:
<RelativeLayout
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"
tools:context=".MainActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="200dp"
android:id="@+id/nameText"
android:background="@drawable/fab_linear_layout">
<TextView
android:text="@string/converter"
android:layout_gravity="bottom"
android:textSize="30sp"
android:textColor="#FFFFFF"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:padding="20dp"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_below="@+id/nameText"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:id="@+id/linearLayout1"
android:layout_centerVertical="true"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true">
<EditText
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:inputType="numberSigned|numberDecimal"
android:hint="@string/celsius"
android:id="@+id/celsius"/>
<TextView
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:hint="@string/fahrenheit"
android:id="@+id/fahrenheitTextView"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:layout_below="@+id/linearLayout1"
android:paddingLeft="20dp"
android:paddingRight="20dp"
android:id="@+id/linearLayout2"
android:layout_centerVertical="true"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true">
<EditText
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:inputType="numberSigned|numberDecimal"
android:hint="@string/fahrenheit"
android:id="@+id/fahrenheit"/>
<TextView
android:layout_width="50dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:hint="@string/celsius"
android:id="@+id/celsiusTextView"/>
</LinearLayout>
从代码可以看出,我想去掉多余的TextView,在EditText中得到答案。
您在错误的视图上设置了文本:
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
而不是 Ft
这是一个 TextView
你应该在 F
上调用 setText
这是 EditText。
看看你的 post 你正在进入一个无限循环,应用崩溃了。由于您在两个 EditText 中都有 onTextChanged 方法,因此当 C 更改文本时,更改 F 会更改 C,因为已更改.... 一次又一次。
做一个简单的测试,如果您从 C 或 F 中删除其中一个方法,您会发现更改正在起作用。
您需要做的是区分用户更改的文本和应用程序自动更改的文本。
我的建议如下
定义一个布尔值来进行微分:
EditText C, F;
TextView Ct, Ft;
private Boolean machine_changed_edittext = false;
不是在每个 onTextChanged 添加以下内容:
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
//We check that the change is human
if (! machine_changed_edittext) {
machine_changed_edittext = true;
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
F.setText(String.valueOf(f));
} else {
F.setText("");
}
//Now as it is finished we declare back the Boolean to the correct value
machine_changed_edittext = false;
}
}
我想我应该做对了,但这只是一个你把布尔值改变的游戏。如果我的想法行不通,请告诉我,我们开始聊天,我们可以调试它。
//未经测试的代码。
如果你只想使用两个EditText。如果您从 onTextChanged
中的其他 EditText 中删除文本观察器,它可能会起作用。
假设 Fahrenheit EditText 的文本观察器称为 fWatcher,Celcius EditText 的文本观察器称为 cWatcher。
想法是,当您在摄氏度编辑文本中修改文本时,删除华氏度编辑文本上的文本观察器,这样它就不会在您编辑文本时触发该监听器。
像这样定义 cWatcher :
TextWatcher cWatcher = new TextWatcher() {
double f,
c;
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
fEditText.removeTextChangedListener(fWatcher);
if (!s.toString().equals("")) {
c = Double.parseDouble(s.toString());
f = (((9 / 5) * c) + 32);
Ft.setText(String.valueOf(f));
} else {
Ft.setText("");
}
fEditText.setTextChangedListener(fWatcher);
}
@Override
public void afterTextChanged(Editable s) {
}
}
同样定义另一个TextWatcher。
实现此目的的另一种方法是检查 EditText 是否具有焦点(Kotlin 代码):
inputOne.addTextChangedListener { editableString->
//do nothing if this EditText is not being edited.
if(!inputOne.hasFocus())
return@addTextChangedListener
//do stuff
}