1. ホーム
  2. javafx

[解決済み] JavaFXです。ChangeListenerの実行

2022-02-18 11:56:41

質問

の中でchangeメソッドを実行することはできますか? ChangeListener を初期化します。 変更があったときだけ実行されるからです。例えば、私は TextField で、空のときに値を0にしたいのですが、以下のようにします。

textField.textProperty().addListener(new ChangeListener<String>() {
        @Override
        public void changed(ObservableValue<? extends String> observable, String oldValue, String newValue) {
            if (!newValue.matches("\\d*")) {
                textField.setText(newValue.replaceAll("[^\\d]", ""));
            }

            if (newValue.isEmpty())
                textField.setText("0");

        }
    });

しかし、アプリケーションの開始時には、変更されたメソッドは呼び出されません。では、この問題を回避するにはどうすればよいのでしょうか?

解決方法は?

プロパティに追加されたリスナーにアクセスする方法がない。明らかに、それへの参照を保持することは可能です。

ChangeListener<String> textFieldListener = (observable, oldValue, newValue) -> {

    if (!newValue.matches("\\d*")) {
        textField.setText(newValue.replaceAll("[^\\d]", ""));
    }

    if (newValue.isEmpty())
        textField.setText("0");

};

textField.textProperty().addListener(textFieldListener);
textFieldListener.changed(null, null, textField.getText());

あるいは、より自然に、実際の機能を別のメソッドに移動させることもできます。

textField.textProperty().addListener((observable, oldValue, newValue) -> vetoTextFieldChanges());
vetoTextFieldChanges();

// ...

private void vetoTextFieldChanges() {
    String newText = textField.getText();
    if (!newText.matches("\\d*")) {
        textField.setText(newText.replaceAll("[^\\d]", ""));
    }

    if (newText.isEmpty())
        textField.setText("0");
}

変更を監視し、ビジネスロジックと矛盾する場合は修正するというアプローチ全体は、あまり満足のいくものではないことに注意してください。例えば、そのプロパティに他のリスナーが登録されている場合、そのリスナーはテキストの中間値(無効な値)を見るかもしれません。これを行うためにサポートされているのは TextFormatter . その TextFormatter は両方とも、テキストに要求された変更を拒否することができます。 以前 textProperty() が更新され、テキストが適切な値に変換されるようにします。つまり、あなたの場合、こうすればいいのです。

UnaryOperator<TextFormatter.Change> filter = change -> {

    // remove any non-digit characters from inserted text:
    if (! change.getText().matches("\\d*")) {
        change.setText(change.getText().replaceAll("[^\\d]", ""));
    }

    // if new text is empty, replace all text with "0":
    if (change.getControlNewText().isEmpty()) {
        change.setRange(0, change.getControlText().length());
        change.setText("0");
    }

    return change ;
};

TextFormatter<Integer> formatter = new TextFormatter<Integer>(new IntegerStringConverter(), 0, filter);
textField.setTextFormatter(formatter);

これで、フォーマッタを使って (数値の) 値を直接取得できるようになりました。 value プロパティで指定し、モデルにバインドする、などです。

// assume model is a data model and valueProperty() returns a Property<Integer>:
formatter.valueProperty().bindBidirectional(model.valueProperty());

このようなバインディングは、モデルが変更されたときにフォーマッタ(そして結果としてテキストフィールド)を更新し、また、モデルの値に応じて初期化することに注意してください(したがって、最初の質問に対する解決策を提供することになるのです)。