我有一个简单的应用程序,它显示从表格中的组合框中选择的项目。但是,当在组合框中选择项目时,剩余项目被过滤以包括所选项目中包括其名称的项目。例如,在以下MCVE中,如果您要选择" Apple"从组合框中,控制列表将被过滤以包含" Apple"和"菠萝"。
在应用滤镜后,有时组合框将重置为不再显示所选项目。 当您选择在结果筛选列表中没有任何其他项目的项目时,会出现此问题。例如,如果您选择" Banana"或"菠萝"从组合框中,组合框将显示提示文本,而不是显示所选项目。
请参阅以下MCVE
Main.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);
}
}
Controller.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;
}
}
Fruit.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;
}
}
sample.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。我担心这个问题的主要原因是因为它允许用户两次选择相同的项目。因此,我可以接受阻止用户复制表中项目的解决方案。
答案 0 :(得分:0)
@James_D帮助我弄清楚我遇到了一个已经在较新版本的Java中修复的错误。但由于我目前无法更新我的Java应用程序版本,我仍然需要找到一种方法来阻止用户将重复的项目添加到列表中。一旦我以这种方式表达问题,很容易确定一个体面的工作。
我修改过滤器以排除已过滤列表中的所选项目。
fruitFilter = selectableFruits -> selectableFruits.getName().toLowerCase().contains(name) && !selectableFruits.getName().toLowerCase().equals(name);
组合框仍然没有显示所选项目(它现在不能,因为它不再在列表中),但用户将无法再选择重复的项目。