JavaFX 将组合框中的字符串字段显示为字符串的枚举(在 TableView 中)

标签 java javafx enums combobox tableview

我的目标是在 tableView 中显示类实例的每个字段。该类有一个 enum 类型的字段,该字段有一个 String 类型的字段。 枚举应作为字符串字段名称显示在组合框中。 当然它也必须是可编辑的。

现在什么不起作用: 仅当单击 ComboBox 时才会显示枚举类的 String 字段,否则它是枚举常量的名称。此外,如果选择了组合框中的另一个枚举,则无法提交它进行编辑。单击 return 不会取消选择组合框,也不会调用 commitEdit 方法。如果选择其他列进行编辑,则尝试的编辑将被取消。

我花了一些努力试图解决这个问题,所以我想也许有人可以帮助我。 由于最初的任务是关于企业软件中更大的类,因此我将其抽象化来提出这个问题。

我知道我可以使列保存 String 类型的枚举,并使其与 MyEnum.values() 和 MyEnum.valueOf() 一起使用,但由于原始类太大而导致性能不佳,因此无法投入生产.

这是我的代码作为示例,如果您不明白问题,只需尝试使用组合框一次,您就会明白。

TableView 类型的类:

public class MyClass {

private MyEnum myEnum;

private String string;

public MyClass(MyEnum myEnum, String string) {
    this.myEnum = myEnum;
    this.string = string;
}

public MyEnum getMyEnum() {
    return myEnum;
}

public void setMyEnum(MyEnum myEnum) {
    this.myEnum = myEnum;
}

public String getString() {
    return string;
}

}

它是枚举字段:

public enum MyEnum {

EnumOne("First Enum"),
EnumTwo("Second Enum");

private String name;

public String getName() {
    return name;
}

private MyEnum(String name) {
    this.name = name;
}

}

外汇应用程序:

public class NewFXMain extends Application {

@Override
public void start(Stage primaryStage) {
    ObservableList<MyClass> items = FXCollections.observableArrayList();
    items.add(new MyClass(MyEnum.EnumOne, "String"));
    TableView<MyClass> table = new TableView(items);
    table.setEditable(true);
    TableColumn<MyClass, MyEnum> myEnumColumn = new TableColumn();
    TableColumn<MyClass, String> stringColumn = new TableColumn();

    stringColumn.setCellFactory(TextFieldTableCell.forTableColumn());
    stringColumn.setCellValueFactory(data -> new ReadOnlyStringWrapper(data.getValue().getString()));



    myEnumColumn.setCellFactory((param) -> new MyEnumComboBoxCell());
    myEnumColumn.setCellValueFactory(data -> new ReadOnlyObjectWrapper(data.getValue().getMyEnum()));
    myEnumColumn.setOnEditCommit(
            event -> {
                event.getRowValue().setMyEnum(event.getNewValue());
                System.out.println(event.getRowValue().getMyEnum());

            });

    table.getColumns().addAll(myEnumColumn, stringColumn);

    StackPane root = new StackPane();
    root.getChildren().add(table);
    Scene scene = new Scene(root, 300, 250);
    primaryStage.setScene(scene);
    primaryStage.show();
}

/**
 * @param args the command line arguments
 */
public static void main(String[] args) {
    launch(args);
}

}

class MyEnumComboBoxCell extends ComboBoxTableCell<MyClass, MyEnum> {

private ComboBox<MyEnum> box;

public MyEnumComboBoxCell() {
    box = new ComboBox<>(FXCollections.observableArrayList(MyEnum.values()));
    box.setCellFactory(new Callback<ListView<MyEnum>, ListCell<MyEnum>>() {
        @Override
        public ListCell<MyEnum> call(ListView<MyEnum> param) {
            return new ListCell<MyEnum>() {
                @Override
                protected void updateItem(MyEnum item, boolean empty) {
                    super.updateItem(item, empty);
                    if ( item != null ) setText(item.getName());
                }

            };
        }
    });
}

@Override
public void startEdit() {
    super.startEdit();
    setGraphic(box);
}

@Override
public void commitEdit(MyEnum newValue) {
    super.commitEdit(newValue);
    if ( newValue != null ) {
        setText(newValue.getName());
        getTableView().getSelectionModel().getSelectedItem().setMyEnum(newValue);
        box.setValue(newValue);
    }
}

@Override
public void updateItem(MyEnum item, boolean empty) {
    super.updateItem(item, empty);
    if ( empty ) {
        setGraphic(null);
    } else {
        setGraphic(null);
        setText(item.getName());
    }
}

}

最佳答案

不要在 updateItem 中设置名称,而是使用 StringConverter,例如:

public class MyEnumConverter extends StringConverter<MyEnum>{

    @Override public String toString(MyEnum enumConstant) {
        return enumConstant.getName();
    }

    @Override public MyEnum fromString(String string) {
        return MyEnum.valueOf(string);
    }
}

然后在单元格的构造函数中:

this.setConverter(new MyEnumConverter());

编辑:您不能@Override所有ComboBoxTableCell的方法,因为它们都按您想要的方式工作。另一方面,您不应该为表格单元格指定自己的组合框,因为它有一个。您只需添加一个StringConverter并设置项目。

你可以这样使用:

myEnumColumn.setCellFactory((param) -> new ComboBoxTableCell<>(new StringConverter<MyEnum>() {
            @Override public String toString(MyEnum object) {
                return object.getName();
            }

            @Override public MyEnum fromString(String string) {
                return MyEnum.valueOf(string);
            }
        }, MyEnum.values()));

如果您愿意,可以像我之前提到的那样为 StringConverter 创建一个单独的类,那么只需:

myEnumColumn.setCellFactory(factory -> new ComboBoxTableCell<>(new MyEnumConverter(), MyEnum.values()));

您还可以删除myEnumColumn.setOnEditCommit

关于JavaFX 将组合框中的字符串字段显示为字符串的枚举(在 TableView 中),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46768511/

相关文章:

java - 如何在javafx中为鼠标单击事件调用相同的方法并输入按键

c++ - 为什么枚举与 Windows 中的位字段不兼容?

java - 可以传递给 ExecutorService 的对象,并且 Runnable 之间具有依赖关系

java - SpringJPA native 查询异常

JavaFX Textfield - 释放向上键时覆盖默认行为

c++ - 让枚举值等同于许多其他值

java - 创建带有计算值的枚举

java.awt.Robot.keyPress 在按下引号键时抛出 IllegalArgumentException

java - Android RecyclerView 项目选择问题

java - 在 JavaFX UI 中显示 littleproxy 日志