我已阅读 API 和示例,但无法理解如何填充表格视图。
假设我有一个带有“名称,值”对的两列字符串数组 (String[][])。我现在只想创建一个表格视图,它在两列中显示数据,在第一列中显示名称,在第二列中显示原始数组中所有行的值。
我尝试了什么?没什么,但似乎您需要为每一列创建可观察列表,将其绑定到各自的列,然后将该列添加到 tableview。但这在我看到的所有示例中都涉及“工厂”,这对我来说是一个陌生的概念。
我猜这很简单,但我无法理解它。请帮忙。
对于表中的每一列,您设置/创建一个Cell Value Factory。每一行都有一个对应的对象tableview.getItems()
。要确定此对象如何跨列显示,每列使用其自己的单元格值工厂。工厂接受对象并返回要显示的值。
由于String[][]
是String[]
's 的数组,我们希望工厂接受 aString[]
并返回String
与其列对应的值。
这是以这种方式创建单元值工厂的示例。它有点冗长,但可以用 lambdas 清理!(参见 Lambda 部分)。
// ---------------------------------------Initialize the data
String[][] data = ...; // Get name/value pairs from somewhere
// ---------------------------------------Setup a TableView with two TableColumns
/* ... Code ... */
// ---------------------------------------Add Cell Value Factories
nameColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
@Override
public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
String[] x = p.getValue();
if (x != null && x.length>0) {
return new SimpleStringProperty(x[0]);
} else {
return new SimpleStringProperty("<no name>");
}
}
});
valueColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
@Override
public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
String[] x = p.getValue();
if (x != null && x.length>1) {
return new SimpleStringProperty(x[1]);
} else {
return new SimpleStringProperty("<no value>");
}
}
});
// ---------------------------------------Add Data to TableView
tableView.getItems().addAll(Arrays.asList(data));
Java 有点冗长,尤其是在使用匿名类时。值得庆幸的是,有 lambdas,它为 Java 带来了一点可读性。这是示例中相同的单元格值工厂,用 lambda 重写。
nameColumn.setCellValueFactory((p)->{
String[] x = p.getValue();
return new SimpleStringProperty(x != null && x.length>0 ? x[0] : "<no name>");
});
valueColumn.setCellValueFactory((p)->{
String[] x = p.getValue();
return new SimpleStringProperty(x != null && x.length>1 ? x[1] : "<no value>");
});
源代码
这是一个以这种方式使用单元值工厂的独立 JavaFX 类。
import java.util.Arrays;
import java.util.List;
import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.value.ObservableValue;
import javafx.scene.Scene;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;
import javafx.util.Callback;
/**
*
* @author nonfrt
*/
public class TableStuff extends Application {
@Override
public void start(Stage primaryStage) {
// Create the data structure
String[][] data = new String[5][2];
data[0] = new String[]{"Jon Skeet","876k"};
data[1] = new String[]{"Darin Dimitrov","670k"};
data[2] = new String[]{"BalusC","660k"};
data[3] = new String[]{"Hans Passant","635k"};
data[4] = new String[]{"Marc Gravell","610k"};
// Create the table and columns
TableView<String[]> tv = new TableView();
TableColumn<String[],String> nameColumn = new TableColumn();
nameColumn.setText("Name Column");
TableColumn<String[],String> valueColumn = new TableColumn();
valueColumn.setText("Value Column");
tv.getColumns().addAll(nameColumn,valueColumn);
// Add cell value factories
// nameColumn.setCellValueFactory((p)->{
// String[] x = p.getValue();
// return new SimpleStringProperty(x != null && x.length>0 ? x[0] : "<no name>");
// });
//
// valueColumn.setCellValueFactory((p)->{
// String[] x = p.getValue();
// return new SimpleStringProperty(x != null && x.length>1 ? x[1] : "<no value>");
// });
nameColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
@Override
public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
String[] x = p.getValue();
if (x != null && x.length>0) {
return new SimpleStringProperty(x[0]);
} else {
return new SimpleStringProperty("<no name>");
}
}
});
valueColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
@Override
public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
String[] x = p.getValue();
if (x != null && x.length>1) {
return new SimpleStringProperty(x[1]);
} else {
return new SimpleStringProperty("<no value>");
}
}
});
// Add Data
tv.getItems().addAll(Arrays.asList(data));
// Finish setting the stage
StackPane root = new StackPane();
root.getChildren().add(tv);
Scene scene = new Scene(root, 300, 250);
primaryStage.setTitle("Cell Value Factory Example");
primaryStage.setScene(scene);
primaryStage.show();
}
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
launch(args);
}
}
是的,这很简单,你这个笨蛋。创建一个行类:
Row(String fName, String fValue){
this.fieldName = new SimpleStringProperty(fName);
this.fieldValue = new SimpleStringProperty(fValue);
}
对于 String-array 中的每一行,您创建一个添加到 observableList 的行对象。
private ObservableList<Row> observableList = FXCollections.observableArrayList(
new Row("The Unfun Cat", "Is a terrible programmer"),
new Row("Stack Overflow", "Rules!");
然后为数组中的两列创建一个表列。
TableColumn columnName = new TableColumn("Name");
columnName.setCellValueFactory(
new PropertyValueFactory<Row,String>("fieldName"));
(与价值相同)
然后使用 tableView.setItems(observableList) 将 observableList 添加到 tableView,最后调用方法 tableView.getColumns.addAll(fieldName,fieldValue);
(这让我想知道如何在不知道原始 String[][] 数组中有多少列的一般情况下这样做?行对象是否可以有一个 ArrayList 来表示任意多个 SimpleStringProperties?你将如何连接它到ValueFactories?)
附言。如果有人创建了一个更具教学意义的示例,我将用“已解决”标记奖励他们的帖子。
您也可以在不知道要向表中添加多少列的情况下尝试使用 map!所以你可以使用 map 键作为 tableview 标题
TableView<Map<String, String>> tableView = new TableView<>();
ArrayList<Map<String, String>> valuesArray = new ArrayList<>();
for(Object object : filter.getEntities()) {
Map<String, String> values = getValuesAsStringsForObject(object);
valuesArray.add(values);
}
for(Object o : filter.getEntities()){
List<String> headers = getHeaders(o);
for (String header :headers){
TableColumn<Map<String, String>, String> tableColumn = new TableColumn<>(header);
tableColumn.setCellValueFactory(param -> new SimpleStringProperty(param.getValue().get(header)));
tableView.getColumns().add(tableColumn);
}
break;
}
tableView.getItems().addAll(valuesArray);
return tableView;