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

希望对以后的观众有所帮助。

纳尔