JavaFX显示枚举,其中Combobox中的字符串字段为String(在TableView中)

时间:2017-10-16 10:47:40

标签: java javafx enums combobox tableview

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

现在什么不起作用: 枚举类的String字段仅在单击ComboBox时显示,否则它是枚举常量的名称。此外,如果选择了组合框中的另一个枚举,则不能将其提交进行编辑。单击return不会取消选择Combobox,也不会调用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;
}

}

这是enum字段:

public enum MyEnum {

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

private String name;

public String getName() {
    return name;
}

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

}

FX App:

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());
    }
}

}

2 个答案:

答案 0 :(得分:1)

不要在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方法,因为所有这些方法都可以按您的方式工作。另一方面,您不应为表格单元格指定自己的ComboBox,因为它有一个。您只需添加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

答案 1 :(得分:0)

非常感谢!实际上,我已经花了一天时间,部分与另一个人在一起,所以我真的很感激! :)

但: 我必须实现setOnEditCommit或者支持tableColumn的myEnum字段不会改变!这一切都有效。没有,只显示更改的内容。

    myEnumColumn.setOnEditCommit(
            event ->
    {
        event.getRowValue().setMyEnum(event.getNewValue());

    });