JavaFX 8 中具有 TextFormatter 和/或 UnaryOperator 的整数数字文本字段

2022-09-02 20:37:14

我正在尝试使用JavaFX 8的TextFormatter为整数创建一个数字文本字段。

一元操作器解决方案:

UnaryOperator<Change> integerFilter = change -> {
    String input = change.getText();
    if (input.matches("[0-9]*")) { 
        return change;
    }
    return null;
};

myNumericField.setTextFormatter(new TextFormatter<String>(integerFilter));

使用 IntegerStringConverter 的解决方案:

myNumericField.setTextFormatter(new TextFormatter<>(new IntegerStringConverter()));  

这两种解决方案都有自己的问题。使用UnaryOperator,我只能按预期输入从0到9的数字,但我还需要输入负值,如“-512”,其中符号只允许在第一个位置。我也不希望像“00016”这样的数字,这仍然是可能的。

IntegerStringConverter方法工作得更好:每个无效数字(如“-16-123”)都不被接受,并且像“0123”这样的数字将转换为“123”。但是,转换仅在提交文本(通过按 Enter)或 TextField 失去焦点时发生。

有没有办法在每次更新 TextField 的值时使用 IntegerStringConverter 强制转换第二个方法?


答案 1

转换器与筛选器不同:转换器指定如何将文本转换为值,筛选器筛选器可进行更改。听起来你在这里想要两者,但你希望过滤器更准确地过滤允许的更改。

我通常发现,如果更改被接受,检查文本的新值最容易。您可以选择性地有一个 ,后跟任意位数的后跟。允许空字符串非常重要,否则用户将无法删除所有内容。-1-9

所以你可能需要这样的东西

UnaryOperator<Change> integerFilter = change -> {
    String newText = change.getControlNewText();
    if (newText.matches("-?([1-9][0-9]*)?")) { 
        return change;
    }
    return null;
};

myNumericField.setTextFormatter(
    new TextFormatter<Integer>(new IntegerStringConverter(), 0, integerFilter));

您甚至可以向过滤器添加更多功能,使其以更智能的方式进行处理,例如:-

UnaryOperator<Change> integerFilter = change -> {
    String newText = change.getControlNewText();
    // if proposed change results in a valid value, return change as-is:
    if (newText.matches("-?([1-9][0-9]*)?")) { 
        return change;
    } else if ("-".equals(change.getText()) ) {

        // if user types or pastes a "-" in middle of current text,
        // toggle sign of value:

        if (change.getControlText().startsWith("-")) {
            // if we currently start with a "-", remove first character:
            change.setText("");
            change.setRange(0, 1);
            // since we're deleting a character instead of adding one,
            // the caret position needs to move back one, instead of 
            // moving forward one, so we modify the proposed change to
            // move the caret two places earlier than the proposed change:
            change.setCaretPosition(change.getCaretPosition()-2);
            change.setAnchor(change.getAnchor()-2);
        } else {
            // otherwise just insert at the beginning of the text:
            change.setRange(0, 0);
        }
        return change ;
    }
    // invalid change, veto it by returning null:
    return null;
};

这将允许用户在任何时候按下,它将切换整数的符号。-

SSCCE:

import java.util.function.UnaryOperator;

import javafx.application.Application;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.TextField;
import javafx.scene.control.TextFormatter;
import javafx.scene.control.TextFormatter.Change;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;
import javafx.util.StringConverter;
import javafx.util.converter.IntegerStringConverter;

public class IntegerFieldExample extends Application {

    @Override
    public void start(Stage primaryStage) {
        TextField integerField = new TextField();
        UnaryOperator<Change> integerFilter = change -> {
            String newText = change.getControlNewText();
            if (newText.matches("-?([1-9][0-9]*)?")) { 
                return change;
            } else if ("-".equals(change.getText()) ) {
                if (change.getControlText().startsWith("-")) {
                    change.setText("");
                    change.setRange(0, 1);
                    change.setCaretPosition(change.getCaretPosition()-2);
                    change.setAnchor(change.getAnchor()-2);
                    return change ;
                } else {
                    change.setRange(0, 0);
                    return change ;
                }
            }
            return null;
        };

        // modified version of standard converter that evaluates an empty string 
        // as zero instead of null:
        StringConverter<Integer> converter = new IntegerStringConverter() {
            @Override
            public Integer fromString(String s) {
                if (s.isEmpty()) return 0 ;
                return super.fromString(s);
            }
        };

        TextFormatter<Integer> textFormatter = 
                new TextFormatter<Integer>(converter, 0, integerFilter);
        integerField.setTextFormatter(textFormatter);

        // demo listener:
        textFormatter.valueProperty().addListener((obs, oldValue, newValue) -> System.out.println(newValue));

        VBox root = new VBox(5, integerField, new Button("Click Me"));
        root.setAlignment(Pos.CENTER);
        Scene scene = new Scene(root, 300, 120);
        primaryStage.setScene(scene);
        primaryStage.show();
    }

    public static void main(String[] args) {
        launch(args);
    }
}

答案 2

有没有办法在每次更新 TextField 的值时使用 IntegerStringConverter 强制转换第二个方法?

虽然@James_D的答案给了你想要的东西,但我也想添加一个不同的视角。你的想法是握住用户的手,每一次按键。这可能会有所帮助,但也可能使用户感到沮丧。将复制粘贴到文本字段中或编辑不同位置的现有输入等操作不适用于手持方法。

您可能希望立即应用转换/筛选的一个原因是,用户可能不知道输入无效,并且在跳转到下一个字段时可能缺少更正。因此,如何不限制用户可以输入的内容,而是可视化当前输入是否有效,而无需更改文本内容。例如,您可以在内容无效时向文本字段添加红色边框。除此之外,您仍然可以使用StringConverter。

例如

myNumericField.setTextFormatter(new TextFormatter<>(new IntegerStringConverter()));
myNumericField.textProperty().addListener((obs,oldv,newv) -> {
    try {
        myNumericField.getTextFormatter().getValueConverter().fromString(newv);
        // no exception above means valid
        myNumericField.setBorder(null);
    } catch (NumberFormatException e) {
        myNumericField.setBorder(new Border(new BorderStroke(Color.RED, BorderStrokeStyle.SOLID, new CornerRadii(3), new BorderWidths(2), new Insets(-2))));
    }
});

text field with red border

转换器也可以很容易地扩展,以限制有效的数字范围。