过滤组合框列表后,JavaFX组合框选定项目消失

时间:2017-07-17 18:34:09

标签: javafx

我有一个简单的应用程序,它显示从表格中的组合框中选择的项目。但是,当在组合框中选择项目时,剩余项目被过滤以包括所选项目中包括其名称的项目。例如,在以下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。我担心这个问题的主要原因是因为它允许用户两次选择相同的项目。因此,我可以接受阻止用户复制表中项目的解决方案。

1 个答案:

答案 0 :(得分:0)

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

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

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

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