2

我试图让 JFXTreeTableView 列显示文本和图像,同时继续使用树分组功能。

我在演示之后对我的程序进行了建模,并且能够使基于 StringProperty 的列正常工作,但现在尝试添加ObjectProperty<Label>列是行不通的。谁能给我一个使用 JFXTreeTableView 和除原始或字符串以外的任何东西(即使用 的东西ObjectProperty<?>)的列示例吗?

编辑:我刚刚在这方面取得了一些进展,方法是使用 HBox 作为我能够显示图像和文本的列。

下面是我用来实现这一目标的代码......

在我的单元格对象中:

public final class TestObj extends RecursiveTreeObject<TestObj> {
        private final ObjectProperty<HBox> source;

        public TestObj(String source) {
                ImageView iv = new ImageView();
                HBox hbox = new HBox();
                hbox.setSpacing(10);
                VBox vbox = new VBox();
                Label l = new Label(source);
                vbox.getChildren().add(l);
                iv.setFitHeight(20);
                iv.setFitWidth(20);
                iv.setImage(new Image("/images/test.png"));
                l.setGraphic(iv);
                hbox.getChildren().addAll(vbox);
                this.source = new SimpleObjectProperty<>(hbox);
        }

        // ... getters & setters
}

在我的控制器中:

public void initialize(URL location, ResourceBundle resources) {
        JFXTreeTableColumn<TestObj, HBox> srcColumn = new JFXTreeTableColumn<>("Source");
        srcColumn.setPrefWidth(150);
        srcColumn.setCellValueFactory((TreeTableColumn.CellDataFeatures<TestObj, HBox> x) ->
        srcColumn.validateValue(x) ? x.getValue().getValue().sourceProperty() : srcColumn.getComputedValue(x));

        //... reset of table setup code
}

现在的问题是列不能正确分组!也就是说:每行的图像和文本可能完全相同,但不会折叠成一个组。相反,它会折叠成许多组,每个组都有一个元素。这让我觉得表格如何比较单元格是问题所在......也许我需要在某个地方覆盖一个方法?

编辑 2:现在,使用下面的代码,可以进行分组并且未分组的节点被正确隐藏。此外,应用 James_D 注释将 GUI 元素从数据类中重新定位。当前的问题(在上面的段落中概述)如图所示。A和图。B.

数据对象:

public final class TestObj extends RecursiveTreeObject<TestObj> {
        //inapplicable vars omitted from this class
        //so don't ask why there's just an inner class here ;)
        private final ObjectProperty<Register> source;
        private final ObjectProperty<Register> destination;

        public TestObj(String src, String srcPort, String srcCountry, String dst, String dstPort, String dstCountry) {
                this.source = new SimpleObjectProperty<>(new Register(src, srcPort, srcCountry));
                this.destination = new SimpleObjectProperty<>(new Register(dst, dstPort, dstCountry));
        }

        // ... getters & setters

        public class Register {
                private final String address;
                private final String port;
                private final String country;

                public Register(String address, String port, String country) {
                    this.address = address;
                    this.port = port;
                    this.country = country;
                }

                //getters ...
         }
}

控制器:

public void initialize(URL location, ResourceBundle resources) {
        JFXTreeTableColumn<TestObj, TestObj.Register> srcColumn = new JFXTreeTableColumn<>("Source");
        srcColumn.setPrefWidth(150); srcColumn.setCellValueFactory((TreeTableColumn.CellDataFeatures<TestObj, TestObj.Register> x) ->
            srcColumn.validateValue(x) ? x.getValue().getValue().sourceProperty() : srcColumn.getComputedValue(x));
    srcColumn.setCellFactory(new Callback<TreeTableColumn<TestObj, TestObj.Register>, TreeTableCell<TestObj, TestObj.Register>>() {
        @Override
        public TreeTableCell<TestObj, TestObj.Register> call(TreeTableColumn<TestObj, TestObj.Register> param) {
            return new JFXTreeTableCell<TestObj, TestObj.Register>() {
                ImageView iv = new ImageView();
                @Override
                protected void updateItem(TestObj.Register item, boolean empty) {
                    if(item != null) {
                        HBox hbox = new HBox();
                        hbox.setSpacing(10);
                        VBox vbox = new VBox();
                        Label l = new Label(item.getAddress() + ":" + item.getPort());
                        vbox.getChildren().add(l);
                        iv.setFitHeight(20);
                        iv.setFitWidth(20);
                        iv.setImage(new Image("/flags/" + item.getCountry() + ".png"));
                        l.setGraphic(iv);
                        hbox.getChildren().addAll(vbox);
                        setGraphic(hbox);
                    } else {
                        setGraphic(null);
                    }
                }
            };
        }
    });
        //... above duplicated for dstColumn
        //... reset of table setup code
}

图 A - 展示表格的初始视图(这里一切正常)

演示表格的初始视图(这里一切正常)

图 B - 显示不正确的分组(应该有 3 个组,每个组都有节点)

显示不正确的分组(应该有 3 个组,每个组都有节点)

4

1 回答 1

1

在对 JFoenix 源代码进行了一些审查之后,我意识到幕后没有神奇的比较器……在我的数据对象中,我只需要重写 equals 和 hashCode 方法来考虑复杂对象的变量。

在这种情况下:

@Override
public boolean equals(Object o) {
    if(o == this)
        return true;
    if(!(o instanceof Register))
        return false;
    Register r = (Register)o;
    return Objects.equals(address, r.address) 
        && Objects.equals(port, r.port) 
        && Objects.equals(country, r.country);
}

@Override
public int hashCode() {
    return Objects.hash(address, port, country);
}
于 2017-12-19T20:43:03.947 回答