我的目标是在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());
}
}
}
答案 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());
});