如何使用 DocumentFilter 设置可以在 Jformattedtextfield 中输入的最大值和最小值?

How to set a Maximum and Minimum value that can be entered in a Jformattedtextfield , using DocumentFilter?

我是 Java 编程的新手。我有一个文档过滤器,它只允许带有“.”的数值和小数。在 JFormattedTextField 中。现在我想在该过滤器中实现一个方法,只允许最大值和最小值,就像我想要输入的数字值一样,它必须在 [0-1] 之间,否则它不应该接受输入的任何内容。

现在我不想用JSpinner,因为从0到1有几百万个小数eg: 0.0001, 0.0012 ...

这是我的 documentFilter 代码:

package javagui.views;
import java.text.NumberFormat;
import java.text.ParseException;
import java.util.Locale;

import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DocumentFilter;
import javax.swing.text.PlainDocument;

public class SeulementNumerique extends PlainDocument  {
    public static final String FLOAT = "0123456789.";
      protected String acceptedChars = null;
      protected boolean negativeAccepted = false;
      public SeulementNumerique() {
        this(FLOAT);
      }
      public SeulementNumerique(String acceptedchars) {  
        acceptedChars = acceptedchars;
      }

      public void setNegativeAccepted(boolean negativeaccepted) {
        if (acceptedChars.equals(FLOAT)) {
          negativeAccepted = negativeaccepted;
          acceptedChars += "-";
        }
      }

      public void insertString(int offset, String str, AttributeSet attr) throws BadLocationException {

        if (str == null)
          return;
        for (int i = 0; i < str.length(); i++) {
          if (acceptedChars.indexOf(str.valueOf(str.charAt(i))) == -1)
            return;
        }
        if (acceptedChars.equals(FLOAT) || (acceptedChars.equals(FLOAT + "-") && negativeAccepted)) {
          if (str.indexOf(".") != -1) {
            if (getText(0, getLength()).indexOf(".") != -1) {
              return;
            }
          }
        }
        if (negativeAccepted && str.indexOf("-") != -1) {
          if (str.indexOf("-") != 0 || offset != 0) {
            return;
          }
        }
        super.insertString(offset, str, attr); 
      }
    }

现在在我使用的主要代码中调用该过滤器:

formattedTextField_5 = new JFormattedTextField();
formattedTextField_5.setDocument(new SeulementNumerique());

是否有任何简单的方法来设置可以在 JFormattedTextField 中输入的最大值和最小值?

JTextField 可与自定义 DocumentFilter 一起使用,检查值是否为数字且是否在指定范围内。下面是一个完成此任务的示例 DocumentFilter:

public class RestrictedNumberDocumentFilter extends DocumentFilter{

    private double min;
    private double max;

    public RestrictedNumberDocumentFilter(double min, double max){
        if ( max < min ){
            double temp = max;
            max = min;
            min = temp;
        }
        this.min = min;
        this.max = max;
    }

    @Override
    public void insertString(FilterBypass fb, int off, String str, AttributeSet attr) throws BadLocationException {
        StringBuilder sb = new StringBuilder(fb.getDocument().getText(0, fb.getDocument().getLength()));
        if ( test(sb.toString()) ){
            fb.insertString(off, str, attr);
        }else{
            //warn
        }

    } 
    @Override
    public void replace(FilterBypass fb, int off, int len, String str, AttributeSet attr)throws BadLocationException{
        StringBuilder sb = new StringBuilder(fb.getDocument().getText(0, fb.getDocument().getLength()));
        sb.replace(off, off+len, str);
        if ( test(sb.toString()) ){
            fb.replace(off, len, str, attr); 
        }else{
            //warn
        }

    }
    /**
     * Sanitized the input
     * @param val
     * @return
     */
    private boolean test(String val){
        try{
            double d = Double.parseDouble(val);
            if ( d >= min && d <= max ){
                return true;
            }
            return false;
        }catch(NumberFormatException e){
            return false;
        }
    }
}

要使用此 class,请为 JTextField 的文档设置 DocumentFilter

JTextField field = new JTextField();
AbstractDocument doc =  (AbstractDocument )field.getDocument();
doc.setDocumentFilter(new RestrictedNumberDocumentFilter (min, max));

您可能需要根据您预期的任何输入进一步自定义此 class。

您可以使用 JFormattedTextField 执行类似的操作,但使用自定义 DocumentFilter 可能会遇到问题,如 here

您可以将 minimummaximum 字段添加到 SeulementNumerique class 并在 super.insertString(offset, str, attr) 语句之前添加以下代码:

try {
    final String text = getText(0, getLength());
    final String newText = text.substring(0, offset) + str + text.substring(offset);
    final double newValue = Double.valueOf(newText);
    if (newValue < minimum || newValue > maximum)
        return;
} catch (final NumberFormatException e) {
    return;
}