我正在尝试扩展JavaFX TableView类,以便可以从字面上传递一个Entity类,它将显示该类的所有保存对象。我可以使它像这样工作,但是我希望我的自定义类足够通用,只接受Class变量并负责其余的工作:
private void showSuppliers()
{
tabPane.getTabs().stream().filter((tab) -> (tab.getText().compareTo("Fournisseurs") == 0)).map((tab) -> (VBox)tab.getContent()).forEachOrdered((vBox) ->
{
TableView<Supplier> table = new TableView<>();
TableColumn nameColumn = new TableColumn("Nom");
nameColumn.setCellValueFactory(new PropertyValueFactory<>("name"));
TableColumn addressColumn = new TableColumn("Addresse");
addressColumn.setCellValueFactory(new PropertyValueFactory<>("address"));
TableColumn contactColumn = new TableColumn("Contact");
contactColumn.setCellValueFactory(new PropertyValueFactory<>("contactPerson"));
TableColumn phoneNumberColumn = new TableColumn("Téléphone");
phoneNumberColumn.setCellValueFactory(new PropertyValueFactory<>("phoneNumber"));
TableColumn emailColumn = new TableColumn("Courriel");
emailColumn.setCellValueFactory(new PropertyValueFactory<>("emailAddress"));
TableColumn creationDateColumn = new TableColumn("Date de création");
creationDateColumn.setCellValueFactory(new PropertyValueFactory<>("enteredDate"));
table.getColumns().addAll(nameColumn, addressColumn, contactColumn, phoneNumberColumn, emailColumn, creationDateColumn);
ObservableList<Supplier> suppliers = FXCollections.observableArrayList(supplierService.findAll());
table.setItems(suppliers);
vBox.getChildren().add(table);
});
}
同样,上述方法可以完美地工作,但是我将不得不为列出我要避免的不同数据的每个表重复该方法。我的计划是使用通用类型扩展TableView,例如:
public class EntityTable<T> extends TableView<T>
{
private final Class<T> entityClass;
public EntityTable(Class<T> entityClass)
{
this.entityClass = entityClass;
init();
}
private void init()
{
for (Field field : entityClass.getDeclaredFields())
{
if ((field.getType() != Set.class) && (field.getName().compareTo("id") != 0))
{
TableColumn column = new TableColumn(field.getName());
column.setCellValueFactory(new PropertyValueFactory<>(field.getName()));
getColumns().add(column);
}
}
}
}
如您所见,添加列不是问题。我不明白的是如何使用我的T来生成类似ObservableList的内容:
ObservableList<Supplier> suppliers = FXCollections.observableArrayList(supplierService.findAll());
我尝试仅将<Supplier>
替换为<T>
或<Class<T>>
,但这似乎无效。当这些东西几乎不存在时,我就回溯到Java了,说实话,我在网上发现的信息有点令人困惑。
如何使用T变量作为类型来构造列表?
谢谢!
答案 0 :(得分:0)
嗯,这很简单,我将其完全删除了!这是最终结果,只需传递一个实体类和一个列表,表将负责其余的工作!
public class EntityTable<T> extends TableView<T>
{
private final Class<T> entityClass;
private final List<T> entityList;
public EntityTable(Class<T> entityClass, List<T> entityList)
{
this.entityClass = entityClass;
this.entityList = entityList;
init();
}
private void init()
{
for (Field field : entityClass.getDeclaredFields())
{
if ((field.getType() != Set.class) && (field.getName().compareTo("id") != 0))
{
TableColumn column = new TableColumn(field.getName());
column.setCellValueFactory(new PropertyValueFactory<>(field.getName()));
getColumns().add(column);
}
}
ObservableList list = FXCollections.observableArrayList(entityList);
setItems(list);
}
}
不利之处在于,它将在类中显示具有实际属性名称的列标题(并不总是非常用户友好)。欢迎提供任何在列标题中自动添加更好内容的建议!
EDIT
对于我的列名,我最终创建了一个仅带有值参数的自定义批注,名为FriendlyName。现在,我像这样填充列标题:
FriendlyName friendlyName = field.getAnnotation(FriendlyName.class);
TableColumn column = new TableColumn(friendlyName.value());