2

在与 Netbeans 和 Scenebuilder 混了一段时间之后,我遇到了一个我不太明白的问题。我使用自定义cellfactory将双击事件绑定到我的表格视图中的单元格。但是当我设置 cellfactorycellValueFactory 时,只有自定义 cellFactory 有效果。

我正在尝试使用来自多个对象的数据填充表格视图,并将双击事件绑定到第一列的单元格。填充不是问题,我只是使用

idNumber.setCellValueFactory(new PropertyValueFactory<LiveStock, String>("idNumber"));
status.setCellValueFactory(new PropertyValueFactory<LiveStock, String>("status"));

然后,我四处搜索以找出如何将双击事件绑定到表格的单元格并找到javafx、TableView:检测单元格上的双击 ...我定义了一个自定义 cellFactory,如下所示:

Callback<TableColumn<LiveStock, String>, TableCell<LiveStock, String>> cellFactory =
    new Callback<TableColumn<LiveStock, String>, TableCell<LiveStock, String>>() {
        @Override
        public TableCell call(TableColumn p) {
            TableCell cell = new TableCell<LiveStock, String>() {};

            cell.addEventFilter(MouseEvent.MOUSE_CLICKED, new EventHandler<MouseEvent>() {
                @Override
                public void handle(MouseEvent event) {
                    if (event.getClickCount() == 2) {
                        System.out.println("double clicked!");
                        TableCell c = (TableCell) event.getSource();
                        System.out.println("Livestock ID: " + c.getId());
                    }
                }
            });

            return cell;
        }

我删除了updateandtoString方法只是为了看看它们是否是我遇到问题的原因。

所以我尝试了

idNumber.setCellFactory(cellFactory);
idNumber.setCellValueFactory(new PropertyValueFactory<LiveStock, String>("idNumber"));

这导致我的单元格为空,但具有双击绑定

有任何想法吗?

我的 LiveStock 类如下所示:

package projekt1.fx;

import javafx.beans.property.SimpleStringProperty;

public class LiveStock {
    private final int idNumber;
    private final SimpleStringProperty ownerID;
    private SimpleStringProperty status;
    private double lat;
    private double longd;


    public LiveStock(int idNumber, String ownerID) {
        this.idNumber = idNumber;
        this.ownerID = new SimpleStringProperty(ownerID);
        this.status = new SimpleStringProperty("ok");
    }

    public int getIdNumber() {
        return this.idNumber;
    }

//    public void setIdNumber(int number) {
//        this.idNumber = number;
//    }

    public String getOwnerID(){
        return ownerID.get();
    }

    public void setOwnerID(String id){
        ownerID.set(id);
    }

    public String getStatus(){
        return status.get();
    }

    public void setStatus(String st){
        status.set(st);
    } 
}

细胞工厂现在看起来像这样:

    Callback<TableColumn<LiveStock, String>, TableCell<LiveStock, String>> cellFactory =
        new Callback<TableColumn<LiveStock, String>, TableCell<LiveStock, String>>() {
            @Override
            public TableCell call(TableColumn p) {
                TableCell cell = new TableCell<LiveStock, String>() {
                    @Override
                    public void updateItem(String item, boolean empty) {
                        super.updateItem(item, empty);
//                        setText("HELLO WORLD!");
                        setText(empty ? null : getString());
                    }
                };

                cell.addEventFilter(MouseEvent.MOUSE_CLICKED, new EventHandler<MouseEvent>() {
                    @Override
                    public void handle(MouseEvent event) {
                        if (event.getClickCount() == 2) {
                            System.out.println("double clicked!");
                            TableCell c = (TableCell) event.getSource();
                            System.out.println("Livestock ID: " + c.getId());
                            togglePopup(null);
                        }
                    }
                });

                return cell;
            }
        };
4

1 回答 1

3

Cell API的文档说:

因为到目前为止,单元格最常见的用例是向用户显示文本,所以此用例专门针对 Cell 内部进行了优化。这是通过 Cell 从Labeled扩展来完成的。这意味着Cell的子类只需要设置 text属性,而不是创建一个单独的 Label 并在 Cell 中设置它。...

Cell 构造函数的当前源代码将文本设置为 null:

public Cell() {
    setText(null);
    ...
}

subclassIndexedCell和 sub-subclass TableCell,它们都没有设置Labeled. 文本在源代码中
默认设置为单元工厂。TableColumn

public static final Callback<TableColumn<?,?>, TableCell<?,?>> DEFAULT_CELL_FACTORY = new Callback<TableColumn<?,?>, TableCell<?,?>>() {
    @Override public TableCell<?,?> call(TableColumn<?,?> param) {
        return new TableCell() {
            @Override protected void updateItem(Object item, boolean empty) {
                if (item == getItem()) return;

                super.updateItem(item, empty);

                if (item == null) {
                    super.setText(null);
                    super.setGraphic(null);
                } else if (item instanceof Node) {
                    super.setText(null);
                    super.setGraphic((Node)item);
                } else {
                    super.setText(item.toString());
                    super.setGraphic(null);
                }
            }
        };
    }
};

但是,通过定义您自己的单元格工厂来创建新TableCell的但未在其覆盖的updateItem()方法中设置文本,将导致一个空 (=null) 列单元格文本。所以是的,问题的原因是删除了内部updateItem调用的方法setText(...)

编辑:
为 TableColumns 明确指定泛型类型,

TableColumn<LiveStock, Integer> idNumber = new TableColumn<LiveStock, Integer>("ID No");

这将避免类型不匹配或错误的类型转换。
然后您的用例的单元工厂回调将是

Callback<TableColumn<LiveStock, Integer>, TableCell<LiveStock, Integer>> cellFactory =
        new Callback<TableColumn<LiveStock, Integer>, TableCell<LiveStock, Integer>>() {
    public TableCell<LiveStock, Integer> call(TableColumn<LiveStock, Integer> p) {
        TableCell<LiveStock, Integer> cell = new TableCell<LiveStock, Integer>() {

            @Override
            public void updateItem(Integer item, boolean empty) {
                super.updateItem(item, empty);
                setText((item == null || empty) ? null : item.toString());
                setGraphic(null);
            }
        };

        cell.addEventFilter(MouseEvent.MOUSE_CLICKED, new EventHandler<MouseEvent>() {
            @Override
            public void handle(MouseEvent event) {
                if (event.getClickCount() > 1) {
                    System.out.println("double clicked!");
                    TableCell c = (TableCell) event.getSource();
                    System.out.println("Cell text: " + c.getText());
                }
            }
        });
        return cell;
    }
};

有什么改变?
LiveStock 中 idNumber 的类型是int. 通过定义new TableColumn<LiveStock, Integer>,我们说这是 LiveStock 行中的一列,其属性 idNumber 具有 int 类型,但泛型类型必须是引用类型,不能TableCell<LiveStock, int>这样我们定义TableCell<LiveStock, Integer>。规则的要点:行项目类的属性类型应该匹配 TableColumn 的第二个泛型类型参数,因此也匹配 TableCell 的参数。

getString 方法在您提到的参考答案链接中定义。但这只是用户定义的方法,并非强制使用。

于 2013-10-08T18:42:43.417 回答