TreeTableColumn.visible:无法设置绑定值

时间:2016-11-16 06:01:30

标签: java javafx-8 treetableview jxtreetable property-binding

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

  

引起:java.lang.RuntimeException:TreeTableColumn.visible:A   无法设置绑定值。

如果我使用双向绑定,我不会收到此错误。但我不需要双向绑定。这是一个错误还是我没有正确使用绑定?请使用以下代码重现此错误。我用的是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>

1 个答案:

答案 0 :(得分:0)

我认为这实际上不是错误。但是,这绝对是神秘而意外的行为。问题的一部分是绑定冲突来自TableHeaderRow,即created by the skin at display time

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

TableView with column header button

结果,TableHeaderRow在这些菜单项的选择状态和每一列的visible属性之间创建了双向绑定。

实际上可以撤消此绑定,但是我发现它很烦人,因为TableHeaderRownull,直到显示TableView为止。但是,采用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的侦听器会在每次显示列时重建菜单并重新链接属性。 / p>

您当然可以找到一种禁用该侦听器的方法。但是显然,这已经很荒谬,而且可能是不必要的。

无论如何,如您所述:

  

如果我使用双向绑定,则不会出现此错误。但是我不需要双向绑定。

我认为第二条语句在技术上并不正确:您的用例不需要双向绑定,但实际上在这里是合适的,因为已经有 个其他属性与列的可见性关联了。 / p>

因此,我将使用双向绑定。