前言
日常开发中,经常需要监听EditText的变化改变TextView。有的时候还有EditText和EditText联动,EditText和SeekBar联动,这种情况要注意避免死循环,关键点是找到一个标识来识别谁影响谁,例如可以获取焦点判断,涉及到的知识点如下:
EditText.addTextChangeListener(...)
EditText.setOnFocusChangeListener(...)
SeekBar.setOnSeekBarChangeListener(...)
相关知识
1.EditText.addTextChangeListener(...)
这个监听有三个方法,大多是在onTextChanged()和afterTextChanged()处理。经过测试单字符输入删除、中间输入删除、多字符输入删除等,发现beforeTextChanged()和onTextChanged()的后面三个参数打印的数据是一一对应的,即start - start,count - before ,after - count。具体参数意思可看注释,此处不举例数据讲解了。输入emoji表情为啥count是2,可以看另一篇文章https://www.jianshu.com/p/9434db689602
mEtPrice.addTextChangedListener(new TextWatcher() {
/**
* 输入框改变前的内容
* @param charSequence 输入前字符串
* @param start 起始光标
* @param count 删除字符串的数量(这里的count是用str.length()计算的,因为删除一个emoji表情,count打印结果是 2)
* @param after 输入框中改变后的字符串与起始位置的偏移量(也就是输入字符串的length)
*/
@Override
public void beforeTextChanged(CharSequence charSequence, int start, int count, int after) {
}
/**
* 输入框改变后的内容
* @param charSequence 字符串信息
* @param start 起始光标
* @param before 输入框中改变前的字符串与起始位置的偏移量(也就是删除字符串的length)
* @param count 输入字符串的数量(输入一个emoji表情,count打印结果是2)
*/
@Override
public void onTextChanged(CharSequence charSequence, int start, int before, int count) {
}
/**
* @param editable 输入结束呈现在输入框中的信息
*/
@Override
public void afterTextChanged(Editable editable) {
}
});
2.EditText.setOnFocusChangeListener(...)
判断EditText是否获取了焦点
mEtValueA.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View view, boolean b) {
focusChangeFlag = b;
}
});
3.SeekBar.setOnSeekBarChangeListener(...)
SeekBar滑动监听,代码调用的setProgress(...)只会执行onProgressChanged(...)这个方法,另外两个方法不执行,且参数fromUser为false。如果是用户手动拖动,三个方法都会执行,且fromUser返回true。
mSeekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
/**
* 进度条改变事件
* @param seekBar
* @param progress 进度条
* @param fromUser 是否是用户操作,用户手动滑动时-true,调用setProgress()-false
*/
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
if(fromUser){ //是用户操作,去影响EditText
mEtFloatRatio.setText(progress+"");
}
}
/**
* 进度条开始拖动事件
* @param seekBar
*/
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
/**
* 进度条停止拖动事件(手指离开屏幕后)
* @param seekBar
*/
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
}
});
实例
image.jpgEditText和EditText联动
获取EditTexti焦点判断,当自身获取到焦点时,才去影响其他EditText,避免死循环。
private boolean AFocusChangeFlag; //valueA焦点状态
private boolean BFocusChangeFlag; //valueB焦点状态
//1.焦点监听事件
mEtValueA.setOnFocusChangeListener((view, b) -> AFocusChangeFlag = b);
mEtValueB.setOnFocusChangeListener((view, b) -> BFocusChangeFlag = b);
//2.mEtValueA.addTextChangedListener(...)的监听
public void afterTextChanged(Editable editable) {
//首位为"" or 0,置1,输入的数字从1开始
if (editable.toString().startsWith("0") || editable.toString().equals("") || editable.toString().startsWith(".")) {
mEtValueA.setText("1");
mEtValueA.setSelection(1);
return;
}
if(AFocusChangeFlag){ //编辑A的情况下,去改变B
mEtValueB.setText(10000 - Integer.valueOf(editable.toString()) +"");
}
}
//3.mEtValueB.addTextChangedListener(...)的监听
@Override
public void afterTextChanged(Editable editable) {
if (editable.toString().startsWith("0") || editable.toString().equals("") || editable.toString().startsWith(".")) {
mEtValueB.setText("1");
mEtValueB.setSelection(1);
return;
}
if(BFocusChangeFlag){ //编辑A的情况下,去改变B
mEtValueA.setText(10000 - Integer.valueOf(editable.toString()) +"");
}
}
EditText和SeekBar联动
两种方式处理:
1.还是获取EditText焦点,但因为拖动SeekBar的时候,EditText没有失去焦点,所以要自己去取消EditText的焦点。
2.判断SeekBar是否是用户手动滑动的,如果是才去影响EditText。(推荐这种)
private int floatRatio; //浮动比例,范围 [0,100]
//1.mEt.addTextChangedListener(...)的方法
@Override
public void afterTextChanged(Editable editable) {
//允许输入的范围是[0,100]
String str = editable.toString();
if (str.startsWith("00") || str.equals("")) {
mEtFloatRatio.setText("0");
mEtFloatRatio.setSelection(1);
return;
}
floatRatio = Integer.valueOf(str);
if (floatRatio>100){
mEtFloatRatio.setText("100");
mEtFloatRatio.setSelection(3);
return;
}
//改变进度条
mSeekBar.setProgress(floatRatio);
}
//2.mSeekBar.setOnSeekBarChangeListener(...)的方法
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
if(fromUser){ //是用户操作,去影响EditText
mEtFloatRatio.setText(progress+"");
}
}
EditText和SeekBar联动,且有+,-按钮
这种情况,不能只用fromUser来判断了,因为点击“+”“-”按钮时,是要SeekBar去影响EditText,而此时是调用的setProgress(),fromUser返回的是false。所以还是用EditText焦点去判断,在用户滑动SeekBar、点击"+","-"时失去焦点。这种处理方式其实用户体验不好,焦点跑别的输入框去了。。。
网友评论