对 javaFX 树排序 table 查看项目
Sort javaFX Tree table view items
我在整理 JFXTreeTableView
(来自 Jfoenix)时遇到问题。
我创建了一个名为 ScanModel 的对象,它扩展了 Jfoenix 的 RecursiveTreeObject<ScanModel>
。
我拥有的众多属性之一是:
private IntegerProperty id;
这是它的 getter 和 setter
public final IntegerProperty idProperty() {
if (id == null) {
id = new SimpleIntegerProperty();
}
return id;
}
public final int getId() {
return idProperty().get();
}
public void setId(int value) {
idProperty().set(value);
}
两个 classes 扩展抽象 ScanModel class:StudyModel 和 SeriesModel。
StudyModel(包含多个 SeriesModel)覆盖了 Jfoenix RecursiveTreeObject 的 getChildren()
方法,以及 returns 其系列可观察列表。
以下是我如何将包含项目的可观察列表绑定到 table 本身:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
tree.setRoot(new RecursiveTreeItem<ScanModel>(elements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
}
到目前为止一切正常,我已经将新的 ScanModel 项目添加到可观察列表中,并且它会自动显示在 UI.
中
现在是我的问题:
客户端请求排序 table:降序 StudyModels 和升序 SeriesModels。
第一次尝试时,我只尝试对 StudyModels 进行排序。
我已将 createTree 方法更改为此实现:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
SortedList<ScanModel> sortedElements = elements.sorted((l, r) -> Integer.compare(l.getId(), r.getId()));
tree.setRoot(new RecursiveTreeItem<ScanModel>(sortedElements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
}
排序后的observable list确实排序了,但是在UI
中没有体现
我在这里错过了什么?
谢谢。
更新
为了以防万一,这里是 table
的 fxml
<JFXTreeTableView fx:id="_treeTableView">
<placeholder>
<Label text="No Normal Studies to display" />
</placeholder>
<columnResizePolicy>
<JFXTreeTableView fx:constant="CONSTRAINED_RESIZE_POLICY" />
</columnResizePolicy>
<columns>
<JFXTreeTableColumn text="ID" sortable="false"
minWidth="150" maxWidth="150">
<cellValueFactory>
<TreeTableErrorIndicationCellFactory value="id" />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Scan Time" sortable="false"
minWidth="140" maxWidth="140">
<cellValueFactory>
<TreeTableErrorIndicationCellFactory value="formattedScanTime" />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Selected" sortable="false"
minWidth="60" maxWidth="60">
<cellValueFactory>
<ScanSelectedValueFactory />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn sortable="false">
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Archived" sortable="false"
style="-fx-alignment: CENTER-RIGHT;" minWidth="60" maxWidth="60">
<cellValueFactory>
<IsArchivedCellFactory />
</cellValueFactory>
</JFXTreeTableColumn>
</columns>
</JFXTreeTableView>
好久没提这个问题了,看到有人看了这个话题。
这是一个答案。当然不是最好的 - 但它有效。
首先,在创建树时 - 需要 setSortPolicy
树。
这是它在我的项目中的样子:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
tree.setRoot(new RecursiveTreeItem<ScanModel>(elements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
// set sort policy for the table in two levels
tree.setSortPolicy(t -> {
Comparator<TreeItem<ScanModel>> studiesComparator = (r1, r2) -> {
try {
int r1Value = Integer.parseInt(r1.getValue().getDisplayName());
int r2Value = Integer.parseInt(r2.getValue().getDisplayName());
return Integer.compare(r2Value, r1Value);
} catch (NumberFormatException e) {
return r2.getValue().getDisplayName().compareToIgnoreCase(r1.getValue().getDisplayName());
}
};
FXCollections.sort(tree.getRoot().getChildren(), studiesComparator);
for (TreeItem<ScanModel> series : tree.getRoot().getChildren()) {
Comparator<TreeItem<ScanModel>> seriesComparator = (r1, r2) -> {
try {
int r1Value = Integer.parseInt(r1.getValue().getDisplayName());
int r2Value = Integer.parseInt(r2.getValue().getDisplayName());
return Integer.compare(r1Value, r2Value);
} catch (NumberFormatException e) {
return r1.getValue().getDisplayName().compareToIgnoreCase(r2.getValue().getDisplayName());
}
};
series.getChildren().sort(seriesComparator);
}
return true;
});
}
其次,对于添加到树中的每个新项目,我调用 sort
方法。
/**
* Sort the table according to the configured sort policy
*
* @param tree
*/
protected void sortTree(JFXTreeTableView<ScanModel> tree) {
// Keep selected row and set it again after sort
TreeItem<ScanModel> selectedRow = tree.getSelectionModel().getSelectedItem();
tree.sort();
tree.getSelectionModel().select(selectedRow);
}
希望对以后的观众有所帮助。
纳尔
我在整理 JFXTreeTableView
(来自 Jfoenix)时遇到问题。
我创建了一个名为 ScanModel 的对象,它扩展了 Jfoenix 的 RecursiveTreeObject<ScanModel>
。
我拥有的众多属性之一是:
private IntegerProperty id;
这是它的 getter 和 setter
public final IntegerProperty idProperty() {
if (id == null) {
id = new SimpleIntegerProperty();
}
return id;
}
public final int getId() {
return idProperty().get();
}
public void setId(int value) {
idProperty().set(value);
}
两个 classes 扩展抽象 ScanModel class:StudyModel 和 SeriesModel。
StudyModel(包含多个 SeriesModel)覆盖了 Jfoenix RecursiveTreeObject 的 getChildren()
方法,以及 returns 其系列可观察列表。
以下是我如何将包含项目的可观察列表绑定到 table 本身:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
tree.setRoot(new RecursiveTreeItem<ScanModel>(elements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
}
到目前为止一切正常,我已经将新的 ScanModel 项目添加到可观察列表中,并且它会自动显示在 UI.
中现在是我的问题:
客户端请求排序 table:降序 StudyModels 和升序 SeriesModels。
第一次尝试时,我只尝试对 StudyModels 进行排序。
我已将 createTree 方法更改为此实现:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
SortedList<ScanModel> sortedElements = elements.sorted((l, r) -> Integer.compare(l.getId(), r.getId()));
tree.setRoot(new RecursiveTreeItem<ScanModel>(sortedElements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
}
排序后的observable list确实排序了,但是在UI
中没有体现
我在这里错过了什么?
谢谢。
更新
为了以防万一,这里是 table
<JFXTreeTableView fx:id="_treeTableView">
<placeholder>
<Label text="No Normal Studies to display" />
</placeholder>
<columnResizePolicy>
<JFXTreeTableView fx:constant="CONSTRAINED_RESIZE_POLICY" />
</columnResizePolicy>
<columns>
<JFXTreeTableColumn text="ID" sortable="false"
minWidth="150" maxWidth="150">
<cellValueFactory>
<TreeTableErrorIndicationCellFactory value="id" />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Scan Time" sortable="false"
minWidth="140" maxWidth="140">
<cellValueFactory>
<TreeTableErrorIndicationCellFactory value="formattedScanTime" />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Selected" sortable="false"
minWidth="60" maxWidth="60">
<cellValueFactory>
<ScanSelectedValueFactory />
</cellValueFactory>
</JFXTreeTableColumn>
<JFXTreeTableColumn sortable="false">
</JFXTreeTableColumn>
<JFXTreeTableColumn text="Archived" sortable="false"
style="-fx-alignment: CENTER-RIGHT;" minWidth="60" maxWidth="60">
<cellValueFactory>
<IsArchivedCellFactory />
</cellValueFactory>
</JFXTreeTableColumn>
</columns>
</JFXTreeTableView>
好久没提这个问题了,看到有人看了这个话题。
这是一个答案。当然不是最好的 - 但它有效。
首先,在创建树时 - 需要 setSortPolicy
树。
这是它在我的项目中的样子:
protected void createTree(JFXTreeTableView<ScanModel> tree, ObservableList<ScanModel> elements) {
tree.setRoot(new RecursiveTreeItem<ScanModel>(elements, RecursiveTreeObject::getChildren));
tree.setShowRoot(false);
tree.getSelectionModel().clearSelection();
// set sort policy for the table in two levels
tree.setSortPolicy(t -> {
Comparator<TreeItem<ScanModel>> studiesComparator = (r1, r2) -> {
try {
int r1Value = Integer.parseInt(r1.getValue().getDisplayName());
int r2Value = Integer.parseInt(r2.getValue().getDisplayName());
return Integer.compare(r2Value, r1Value);
} catch (NumberFormatException e) {
return r2.getValue().getDisplayName().compareToIgnoreCase(r1.getValue().getDisplayName());
}
};
FXCollections.sort(tree.getRoot().getChildren(), studiesComparator);
for (TreeItem<ScanModel> series : tree.getRoot().getChildren()) {
Comparator<TreeItem<ScanModel>> seriesComparator = (r1, r2) -> {
try {
int r1Value = Integer.parseInt(r1.getValue().getDisplayName());
int r2Value = Integer.parseInt(r2.getValue().getDisplayName());
return Integer.compare(r1Value, r2Value);
} catch (NumberFormatException e) {
return r1.getValue().getDisplayName().compareToIgnoreCase(r2.getValue().getDisplayName());
}
};
series.getChildren().sort(seriesComparator);
}
return true;
});
}
其次,对于添加到树中的每个新项目,我调用 sort
方法。
/**
* Sort the table according to the configured sort policy
*
* @param tree
*/
protected void sortTree(JFXTreeTableView<ScanModel> tree) {
// Keep selected row and set it again after sort
TreeItem<ScanModel> selectedRow = tree.getSelectionModel().getSelectedItem();
tree.sort();
tree.getSelectionModel().select(selectedRow);
}
希望对以后的观众有所帮助。
纳尔