筛选组合框列表后,JavaFX 组合框选定项将消失



我有一个简单的应用程序,它显示从表格的组合框中选择的项目。但是,在组合框中选择项目时,将筛选其余项目以包括其名称包含在所选项目中的项目。例如,在以下 MCVE 中,如果要从组合框中选择"Apple",则控制列表将被筛选为包含"Apple"和"Pineapple"。

有时,应用筛选器后,组合框将重置为不再显示所选项目。 当您选择的项在生成的筛选列表中没有任何其他项时,会出现此问题。例如,如果从组合框中选择"香蕉"或"菠萝",则组合框将显示提示文本,而不是显示所选项目。

请参阅以下 MCVE

主.java

package sample;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
public class Main extends Application {
@Override
public void start(Stage primaryStage) throws Exception{
Parent root = FXMLLoader.load(getClass().getResource("sample.fxml"));
primaryStage.setTitle("ComboBox Issues");
primaryStage.setScene(new Scene(root, 300, 275));
primaryStage.show();
}

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

控制器.java

package sample;
import javafx.application.Platform;
import javafx.beans.binding.Bindings;
import javafx.beans.property.ReadOnlyStringWrapper;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.collections.transformation.FilteredList;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.scene.control.ComboBox;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.util.StringConverter;
import java.util.function.Predicate;
public class Controller {
@FXML
private TableView<Fruit> fruityTable;
@FXML
private ComboBox<Fruit> fruitSelector;
private ObservableList<Fruit> selectedFruits;
private ObservableList<Predicate<Fruit>> filterCriteria;
private Predicate<Fruit> fruitFilter;
@FXML
private TableColumn<Fruit, String> fruitNameColumn;
@FXML
private TableColumn<Fruit, String> fruitColorColumn;
@FXML
void addSelectedFruit(ActionEvent event) {
if (fruitSelector.getValue() != null) {
Fruit selectedFruit = getSelectedFruitFromComboBox();
final String name = selectedFruit.getName().toLowerCase();
fruitFilter = selectableFruits -> selectableFruits.getName().toLowerCase().contains(name);
Platform.runLater(() -> filterCriteria.add(fruitFilter));
this.selectedFruits.add(selectedFruit);
event.consume();
}
}
private Fruit getSelectedFruitFromComboBox() {
return fruitSelector.getValue();
}
@FXML
void initialize() {
Fruit apple = new Fruit("Apple", "Red");
Fruit pineapple = new Fruit("Pineapple", "Brown");
Fruit banana = new Fruit("Banana", "Yellow");
ObservableList<Fruit> fruitSelectorItems = FXCollections.observableArrayList();
fruitSelectorItems.addAll(apple, pineapple, banana);
initializeFruitSelector(fruitSelectorItems);
initializeFruitTable();
}
private void initializeFruitSelector(ObservableList<Fruit> fruitSelectorItems) {
FilteredList<Fruit> filteredFruit = new FilteredList<>(fruitSelectorItems, x -> true);
fruitSelector.setItems(filteredFruit);
filterCriteria = FXCollections.observableArrayList();
filteredFruit.predicateProperty().bind(Bindings.createObjectBinding(() ->
filterCriteria.stream().reduce(x-> true, Predicate::and), filterCriteria));
fruitSelector.setConverter(createFruitChooserConverter());
}
private StringConverter<Fruit> createFruitChooserConverter() {
return new StringConverter<Fruit>() {
@Override
public String toString(Fruit item) {
if (item == null ) {
return null;
} else {
return item.getName();
}
}
@Override
public Fruit fromString(String string) {
return null;
}
};
}
private void initializeFruitTable() {
selectedFruits = FXCollections.observableArrayList();
fruitNameColumn.setCellValueFactory(cellData -> formatFruitNameColumnText(cellData.getValue()));
fruitColorColumn.setCellValueFactory(cellData -> formatFruitColorColumnText(cellData.getValue()));
fruityTable.setItems(selectedFruits);
}
private ObservableValue<String> formatFruitColorColumnText(Fruit fruit) {
ReadOnlyStringWrapper color;
if (fruit == null) {
color = null;
} else {
color = new ReadOnlyStringWrapper(fruit.getColor());
}
return color;
}
private ObservableValue<String> formatFruitNameColumnText(Fruit fruit) {
ReadOnlyStringWrapper name;
if (fruit == null) {
name = null;
} else {
name = new ReadOnlyStringWrapper(fruit.getName());
}
return name;
}
}

水果.java

package sample;
public class Fruit {
private String name;
private String color;
Fruit(String name, String color){
this.name = name;
this.color = color;
}
public String getColor() {
return color;
}
public String getName() {
return name;
}
}

样本.fxml

<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.ComboBox?>
<?import javafx.scene.control.TableColumn?>
<?import javafx.scene.control.TableView?>
<?import javafx.scene.layout.ColumnConstraints?>
<?import javafx.scene.layout.GridPane?>
<?import javafx.scene.layout.RowConstraints?>
<GridPane alignment="center" hgap="10" vgap="10" xmlns="http://javafx.com/javafx/8.0.111" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
<children>
<TableView fx:id="fruityTable" prefHeight="200.0" prefWidth="201.0" GridPane.columnIndex="1">
<columns>
<TableColumn fx:id="fruitNameColumn" prefWidth="75.0" text="Name" />
<TableColumn fx:id="fruitColorColumn" prefWidth="75.0" text="Color" />
</columns>
</TableView>
<ComboBox fx:id="fruitSelector" onAction="#addSelectedFruit" prefWidth="150.0" promptText="Choose a fruit" />
</children>
<columnConstraints>
<ColumnConstraints minWidth="10.0" prefWidth="100.0" />
<ColumnConstraints />
</columnConstraints>
<rowConstraints>
<RowConstraints />
</rowConstraints>
</GridPane>

这似乎是 JavaFX 组合框的一个错误,但我还没有看到有人遇到类似的问题(也许是因为在选择后过滤相同的组合框是不常见的要求?还是我做错了什么?

编辑

正如James_D注释中指出的那样,此问题在较新版本的Java(至少Java 8u131(中不存在。但是,我现在被迫使用Java 8u25。我关注此问题的主要原因是因为它允许用户两次选择同一项目。因此,防止用户复制表中项目的解决方案对我来说是可以接受的。

@James_D帮助我弄清楚我遇到了一个已在较新版本的 Java 中修复的错误。但是由于我目前无法更新我的 Java 应用程序版本,我仍然需要找到一种方法来防止用户能够将重复项添加到列表中。一旦我以这种方式表达问题,就很容易确定一个体面的工作。

我修改了过滤器以排除过滤列表中的选定项目。

fruitFilter = selectableFruits -> selectableFruits.getName().toLowerCase().contains(name) && !selectableFruits.getName().toLowerCase().equals(name);

组合框仍然不显示所选项目(现在不能显示,因为它不再在列表中(,但用户将无法再选择重复的项目。

最新更新