繁体   English   中英

JavaFx ComboBox 绑定混淆

[英]JavaFx ComboBox binding confusion

我有一个 I18N 实现,它通过属性绑定 JavaFX UI 元素,例如:

def translateLabel(l: Label, key: String, args: Any*): Unit =
    l.textProperty().bind(createStringBinding(key, args))

拥有一个属性绑定很容易并且效果很好。 但是我与 ComboBox 斗争,因为它需要一个 ObservableList (在我的情况下是字符串),我不知道如何将我的翻译函数绑定到它。 我对ObservableValueObservableListProperty接口之间的区别感到矛盾,因为它们听起来都一样。

它有itemsProperty()valueProperty()但是这些文档缺乏且含糊不清,所以我不确定它们可以在哪里使用。

我想要做的是有一个 ComboBox ,其中所有元素(或至少选定/可见的元素)动态更改语言(I18N),就像它被绑定一样,就像一个属性一样。

编辑:

为了更容易理解,我目前的实现是:

private def setAggregatorComboBox(a: Any): Unit = {

    val items: ObservableList[String] = FXCollections.observableArrayList(
        noneOptionText.getValue,
        "COUNT()",
        "AVG()",
        "SUM()"
    )

    measureAggregatorComboBox.getItems.clear()

    measureAggregatorComboBox.getItems.addAll(items)
}

其中noneOptionText是已绑定到StringBindingStringProperty ,该 StringBinding 在 class 实例化时以这种方式进行转换:

def translateString(sp: StringProperty, key: String, args: Any*): Unit =
        sp.bind(createStringBinding(key, args))

itemsProperty()要在组合框弹出窗口中显示的项目列表 它的值是ObservableList

valueProperty()选定项(如果组合框可编辑,则为用户输入的值)。

我建议将组合框中的数据作为键列表,并使用自定义单元格将每个单元格中的文本绑定到这些键的翻译。 我不会说 scala,但在 Java 中它看起来像:

ComboBox<String> comboBox = new ComboBox<>();
comboBox.getItems().setAll(getAllKeys());

class TranslationCell extends ListCell<String> {

    @Override
    protected void updateItem(String item, boolean empty) {
        super.updateItem(item, empty);
        textProperty().unbind();
        if (empty || item == null) {
            setText("");
        } else {
            textProperty().bind(createStringBinding(item));
        }
    }
}

comboBox.setCellFactory(lv -> new TranslationCell());
comboBox.setButtonCell(new TranslationCell());

现在请注意, valueProperty()包含所选值的

如果您真的想将项目绑定到ObservableValue<ObservableList<String>>您可以执行以下操作:

comboBox.itemsProperty().bind(Bindings.createObjectBinding(() ->
    FXCollections.observableArrayList(...),
    ...));

其中第一个...String值的可变参数,第二个...是可观察值,其中的更改将提示重新计算列表。 (所以在你的情况下,我猜你在某处有一个ObservableValue<Locale>代表当前语言环境;你会将它用于第二个参数。)

在您的特定用例中(只有列表的第一个元素是可国际化的),使用侦听器可能更容易:

comboBox.getItems().setAll(
    noneOptionTest.getValue(), 
    "COUNT()",
    "AVG()",
    "SUM");
noneOptionTest.addListener((obs, oldVal, newVal) ->
    comboBox.getItems().set(0, newVal));

虽然我同意这稍微不那么优雅。

为了完整性:

我对ObservableValueObservableListProperty接口之间的区别感到矛盾,因为它们听起来都一样。

ObservableValue<T> :表示可以观察到T类型的单个值(这意味着代码可以在更改时执行)。

Property<T> :表示可写ObservableValue<T> 目的是实现将有一个表示该值的实际变量。 它定义了允许其值绑定到其他ObservableValue<T>的附加功能。

因此,例如:

DoubleProperty x = new SimpleDoubleProperty(6);
DoubleProperty y = new SimpleDoubleProperty(9);
ObservableValue<Number> product = x.multiply(y);

xy都是Property<Number> SimpleDoubleProperty的实现有一个代表这个值的实际double变量,你可以做类似y.set(7); 更改值。

另一方面, product不是Property<Number> 您不能更改其值(因为这样做会违反绑定:声明的不变量product.getValue() == x.getValue() * y.getValue() ); 但是它是observable ,所以你可以绑定到它:

BooleanProperty answerCorrect = new SimpleBooleanProperty();
answerCorrect.bind(product.isEqualTo(42));

等等

一个ObservableList有点不同:它是一个java.util.List (元素的集合),你可以观察它来响应对列表的操作 即,如果您将侦听器添加到ObservableList ,侦听器可以确定是否添加或删除了元素等。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM