java - 将自定义 FXML 属性设置为自定义 javafx 组件的参数

标签 java javafx fxml

我创建了自定义组件TableBlock。它由 Label 和 TableView 组成。例如,TableView 可以具有 1 到 1000 行。行数由 FXML 文件中的参数“rowsFromPrefs”定义。创建 TableView 时需要此参数。 TableView完全由JAva代码创建,在fxml中只是它的标签和带有多行的参数。

据我所知,当JavaFX构造FXML组件时,它首先调用构造函数,然后调用@FXML注释的字段,然后启动initialize()方法。

在我的例子中,当initialize()启动时,变量rowsFromPrefs仍然为空!但是,如果我尝试从其他线程(不是 JavaFX 启动器)获取 rowsFromPrefs 的值,我会看到它定义为 =“2”,就像它应该的那样。

所以我无法理解 Java 在什么时候从 FXML 文件分配对象参数。创建对象时如何将参数从 fxml 文件传递​​到对象。

我看到构造函数参数的 @NamedArg 注释。这是创建对象时传递参数的唯一一种方式吗?

the controller can define an initialize() method, which will be called once on >an implementing controller when the contents of its associated document have >been completely loaded:

TableBlock.java

public class TableBlock extends VBox{
    @FXML
    private String rowsFromPrefs;
    @FXML
    private Label label;

public TableBlock() {
    FXMLLoader fxmlLoader = new   FXMLLoader(getClass().getResource("TableBlock.fxml"));
    fxmlLoader.setRoot(this);
    fxmlLoader.setController(this);
    try {
        fxmlLoader.load();
    } catch (IOException e) {
        e.printStackTrace();
    }
}

@FXML
public void initialize() {
    this.table = createTable(rowsFromPrefs);
}

public String getRowsFromPrefs() {
    System.out.println("getRowsFromPrefs");
    return rowsFromPrefs;
}


public void setRowsFromPrefs(String rowsFromPrefs) {
    this.rowsFromPrefs = rowsFromPrefs;
}

}

TableBlock.fxml

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

<?import javafx.scene.control.*?>
<?import ru.laz.model.controls.*?>
<?import java.lang.*?>
<?import javafx.scene.layout.*?>
<?import javafx.scene.layout.AnchorPane?>
<?import ru.laz.model.controls.tableblock.*?>


<fx:root type="javafx.scene.layout.VBox" xmlns="http://javafx.com/javafx/8" xmlns:fx="http://javafx.com/fxml/1">
   <children>
      <Label text="Label" />
   </children>
</fx:root>

View.java

public class View extends Application {
Parent root = null;
private Scene scene;

@Override
    public void init() {
    try {
            root = FXMLLoader.load(getClass().getResource("View.fxml"));
            root.requestLayout();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
}

@Override
    public void start(final Stage stage) throws Exception {
     scene = new Scene(root, 640, 480, Color.LIGHTGRAY);
     stage.show();
}

    public static void main(String[] args) {
       launch(args);
    }

}

查看.fxml

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

<?import java.lang.*?>
<?import javafx.scene.layout.*?>
<?import javafx.scene.control.*?>
<?import ru.laz.model.controls.tableblock.*?>


<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8" xmlns:fx="http://javafx.com/fxml/1">
   <children>
      <TableBlock rowsFromPrefs="2" id="IDDQD"/>
   </children>
</AnchorPane>

最佳答案

首先,请注意 @FXML rowsFromPrefs 上的注释毫无意义。 @FXML当当前对象作为 Controller 的 FXML 文件具有带有 fx:id 的元素时,会导致为字段注入(inject)一个值。其值与字段名称匹配的属性。自 TableBlock.fxml没有包含 fx:id="rowsFromPrefs" 的元素,这个注释没有做任何事情。

FXMLLoader正在加载View.fxml遇到<TableBlock>元素,它创建一个 TableBlock通过调用其构造函数来实例化。然后它将设置属性指定的值。所以你的 FXML 元素

<TableBlock rowsFromPrefs="2" id="IDDQD"/>

本质上等同于

TableBlock tableBlock = new TableBlock();
tableBlock.setRowsFromPrefs("2");
tableBlock.setId("IDDQD");

当然是 TableBlock 的构造函数只是执行代码所说的操作:它创建一个 FXMLLoader ,设置该 FXMLLoader 的根目录和 Controller ,然后调用load()that FXMLLoader 的加载过程将设置@FXML -在 Controller 上注入(inject)字段(正在执行构造函数的 TableBlock 对象),然后调用 initialize() .

所以initialize()作为 FXMLLoader.load() 调用的一部分被调用那是在 TableBlock构造函数;当然这一切都发生在setRowsFromPrefs("2");之前被调用。

总而言之,TableBlock.initialize()TableBlock.fxml 之后调用已被解析,并且其中定义的任何元素都被注入(inject)到其相应的 @FXML 中- 带注释的字段,但这发生在 View.fxml 之前已加载。

解决此问题的一种方法是传递 rowsFromPrefsTableBlock构造函数。为此,请使用 @NamedArg annotation :

public class TableBlock extends VBox{

    private final String rowsFromPrefs;

    @FXML
    private Label label;

    public TableBlock(@NamedArg("rowsFromPrefs") String rowsFromPrefs) {

        this.rowsFromPrefs = rowsFromPrefs ;
        FXMLLoader fxmlLoader = new   FXMLLoader(getClass().getResource("TableBlock.fxml"));
        fxmlLoader.setRoot(this);
        fxmlLoader.setController(this);
        try {
            fxmlLoader.load();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    @FXML
    public void initialize() {
        this.table = createTable(rowsFromPrefs);
    }

    public String getRowsFromPrefs() {
        System.out.println("getRowsFromPrefs");
        return rowsFromPrefs;
    }


}

现在,FXML 中的属性将传递给构造函数而不是 set 方法,因此 rowsFromPrefs将在您调用 fxmlLoader.load() 之前初始化,根据需要。

当然,另一个选择就是将代码从 initialize() 中移出。方法setRowsFromPrefs(...)方法。如果您打算的话,我会使用上述选项 rowsFromPrefs为每个 TableBlock 进行固定实例,并且仅当您希望能够更改 rowsFromBlocks 时才使用第二个选项在个人的生命周期中TableBlock实例。

关于java - 将自定义 FXML 属性设置为自定义 javafx 组件的参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41595035/

相关文章:

Java 服务器无法正常关闭?

java - PreparedStatement 批处理和 Statement 批处理之间的区别

java - 二维数组上的用户输入 (Java)

Eclipse 对 jfxrt.jar 类的访问限制发出警告

java - 如何从 JavaFX2 的父 Controller 获取 Controller 的特定实例?

java - 在 FXML 中设置节点layoutProperty()

java - 如何暂停线程直到处理完数据?发送信号量?

java - 如何在 JavaFX 中使用时间线进行双击?

listview - 如何使ListView中的文本居中?

css - fxml中的旋转图像动画