java - TreeTableColumn.visible : A bound value cannot be set

标签 java javafx-8 treetableview jxtreetable property-binding

我制作了一个简单的 JavaFX 应用程序。在此应用程序中,有一个包含 2 列和一个复选框的树表。如果复选框被选中,第 2 列将可见,否则不可见。为此,我将树表列的可见属性绑定(bind)到复选框选定的属性。当我单击复选框列状态更改但同时给出。

Caused by: java.lang.RuntimeException: TreeTableColumn.visible : A bound value cannot be set.

如果我使用双向绑定(bind),我不会收到此错误。但我不需要双向绑定(bind)。是错误还是我没有正确使用绑定(bind)?请使用以下代码重现此错误。我用的是jdk1.8.0_111。

JavaFXApplication.java

package test;

import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;

public class JavaFXApplication extends Application {

    @Override
    public void start(Stage stage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("FXMLDocument.fxml"));

        Scene scene = new Scene(root);

        stage.setScene(scene);
        stage.show();
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        launch(args);
    }

}

FXMLDocumentController.java

package test;

import java.net.URL;
import java.util.ResourceBundle;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.CheckBox;
import javafx.scene.control.Label;
import javafx.scene.control.TreeTableColumn;
import javafx.scene.control.TreeTableView;

public class FXMLDocumentController implements Initializable {

    @FXML
    private TreeTableView<?> table;
    @FXML
    private CheckBox box;
    @FXML
    private TreeTableColumn<?, ?> c2;

    @Override
    public void initialize(URL url, ResourceBundle rb) {
        // TODO
        c2.visibleProperty().bind(box.selectedProperty());
    }    

}

FXMLDocument.fxml

<?xml version="1.0" encoding="UTF-8"?>

<?import javafx.scene.control.CheckBox?>
<?import javafx.scene.control.TreeTableColumn?>
<?import javafx.scene.control.TreeTableView?>
<?import javafx.scene.layout.AnchorPane?>

<AnchorPane id="AnchorPane" prefHeight="390.0" prefWidth="452.0" xmlns:fx="http://javafx.com/fxml/1" xmlns="http://javafx.com/javafx/8.0.65" fx:controller="javafxapplication2.FXMLDocumentController">
    <children>
      <TreeTableView fx:id="table" layoutX="2.0" prefHeight="390.0" prefWidth="149.0">
        <columns>
          <TreeTableColumn prefWidth="75.0" text="C1" />
          <TreeTableColumn fx:id="c2" prefWidth="75.0" text="C2" visible="false" />
        </columns>
      </TreeTableView>
      <CheckBox fx:id="box" layoutX="234.0" layoutY="77.0" mnemonicParsing="false" text="CheckBox" />
    </children>
</AnchorPane>

最佳答案

我认为这实际上不是错误。不过,这绝对是神秘而出乎意料的行为。部分问题是绑定(bind)冲突来自 TableHeaderRow ,即 created by the skin at display time .

TableHeaderRow 负责呈现所有列标​​题,它包含一个内置菜单按钮,默认情况下,它是要显示/隐藏的列的单选列表:

TableView with column header button

因此,TableHeaderRow 在这些菜单条目的选择状态和每列的可见属性之间创建了双向绑定(bind)。

实际上可以撤消此绑定(bind),但我发现这很烦人,因为在显示 TableView 之前 TableHeaderRownull。但是,适应this solution to access the TableHeaderRow ,我们可以这样做:

      TableHeaderRow headerRow = ((TableViewSkinBase) tableView.getSkin()).getTableHeaderRow();
      try {

        // get columnPopupMenu field
        Field privateContextMenuField = TableHeaderRow.class.getDeclaredField("columnPopupMenu");

        // make field public
        privateContextMenuField.setAccessible(true);

        // get context menu
        ContextMenu contextMenu = (ContextMenu) privateContextMenuField.get(headerRow);

        for (MenuItem menuItem : contextMenu.getItems()) {
          // Assuming these will be CheckMenuItems in the default implementation
          BooleanProperty selectedProperty = ((CheckMenuItem) menuItem).selectedProperty();

          // In theory these menu items are in parallel with the columns, but I just brute forced it to test
          for (TableColumn<?, ?> tableColumn : tableView.getColumns()) {
            // Unlink the column's visibility with the menu item
            tableColumn.visibleProperty().unbindBidirectional(selectedProperty);
          }
        }

      } catch (Exception ex) {
        ex.printStackTrace();
      }

      // Not strictly necessary but we don't want to be misleading :-p
      tableView.setTableMenuButtonVisible(false);

但关键在于:每次列的可见性发生变化时都必须执行此操作,因为 TableHeaderRow 有一个监听器,它会重建菜单并重新链接属性,每次列是显示。

当然,您可以找到一种方法来禁用该监听器..但显然这已经很荒谬而且可能没有必要。

无论如何,正如您所说:

If I use bidirectional binding I don't get this error. But I don't need bidirectional binding.

我认为第二个陈述在技术上是不正确的:您的用例不需要双向绑定(bind),但它实际上适用于此,因为 其他属性已经与列的可见性相关联。

所以,我会使用双向绑定(bind)。

关于java - TreeTableColumn.visible : A bound value cannot be set,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40625086/

相关文章:

JavaFX 8 : shifting table cells when using borders for table row

java - BigInteger 除法() 方法不起作用?

java - Linux 上的 JOGL 没有 glcontext 和 XInitThreads()

java - Spring mvc 使用 @RequestBody 验证原语不起作用

javax.net.ssl.SSLHandshakeException : Remote host closed connection during handshake while load Test using JMeter

java - 如何修改 TreeTableView 的点击行为

java - 如何在JavaFx中获取折线图刻度标签的宽度?

java - 有没有更简单(更轻松)的方法来使用 JavaFX 8 将文本居中放置在 "zone"中?

JavaFx 在表列上使用 String 和 Double

JavaFX TreeTableView 列 setOnEditCommit() 拒绝接受处理程序以保存已编辑的单元格