美文网首页
EditText和EditText的联动、EditText和Se

EditText和EditText的联动、EditText和Se

作者: Dengszzzzz | 来源:发表于2019-07-05 14:53 被阅读0次

    前言

    日常开发中,经常需要监听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.jpg
    EditText和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、点击"+","-"时失去焦点。这种处理方式其实用户体验不好,焦点跑别的输入框去了。。。

    相关文章

      网友评论

          本文标题:EditText和EditText的联动、EditText和Se

          本文链接:https://www.haomeiwen.com/subject/lvoshctx.html