Java 无法使用@FXML ANOTION访问超类控制器中的元素
我正在尝试使用JavaFX为我的程序建立一个简单的示例。 我想要的是一个带有main.fxml的控制器;然后main.fxml将有一个TabbedPane作为根目录,有两个选项卡(tab1.fxml和tab2.fxml),每个选项卡都有其控制器(Tab1Controller和Tab2Controller) 这可能是个问题,但我不明白为什么会有问题: 选项卡1控制器和选项卡2控制器都扩展控制器;因为它们共享各种被操纵的字段,例如需要不断更新的底部状态栏 到目前为止,我设法使所有控制器和.fxml都正常工作 当我试图从其中一个子类在控制器中设置标签的文本时,它只指向null,但标签在gui中以其默认文本初始化。。 编译正确,并且lblController似乎正确链接到main.fxml中的fx:id 非常感谢任何帮助/链接 我一直在看各种帖子,但唯一一篇接近我需要的帖子是这篇: 主要内容: 主控制器:Java 无法使用@FXML ANOTION访问超类控制器中的元素,java,javafx,fxml,Java,Javafx,Fxml,我正在尝试使用JavaFX为我的程序建立一个简单的示例。 我想要的是一个带有main.fxml的控制器;然后main.fxml将有一个TabbedPane作为根目录,有两个选项卡(tab1.fxml和tab2.fxml),每个选项卡都有其控制器(Tab1Controller和Tab2Controller) 这可能是个问题,但我不明白为什么会有问题: 选项卡1控制器和选项卡2控制器都扩展控制器;因为它们共享各种被操纵的字段,例如需要不断更新的底部状态栏 到目前为止,我设法使所有控制器和.fxml都
package sample;
import javafx.fxml.FXML;
import javafx.scene.control.Label;
public class Controller {
@FXML
public Label lblController;
public Controller() {
System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
}
@FXML
public void initialize() {
System.out.println("CONTROLLER MAIN - INITIALIZER");
}
protected void setSts(String sts) {
lblController.setText(sts);
}
}
main.fxml
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.Tab?>
<?import javafx.scene.control.TabPane?>
<?import javafx.scene.layout.*?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
xmlns="http://javafx.com/javafx/8.0.60"
fx:controller="sample.Controller">
<BorderPane>
<center>
<TabPane tabClosingPolicy="UNAVAILABLE">
<Tab text="Tab 1">
<AnchorPane>
<!--FOR fx:id HAS TO HAVE THE SAME NAME BUT IN LOWER CASE AS THE .fxml-->
<fx:include fx:id="tab1" source="tab/tab1.fxml"/>
</AnchorPane>
</Tab>
<Tab text="Tab 2">
<AnchorPane>
<fx:include fx:id="tab2" source="tab/tab2.fxml"/>
</AnchorPane>
</Tab>
</TabPane>
</center>
<bottom>
<AnchorPane>
<Label fx:id="lblController" text="Controller Test"/>
</AnchorPane>
</bottom>
</BorderPane>
</AnchorPane>
tab1.fxml
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
xmlns="http://javafx.com/javafx/8.0.60" fx:controller="sample.ctrTab.Tab1Controller">
<children>
<Label fx:id="lbl1" layoutX="100" layoutY="50" text="Def tab 1"/>
<TextField fx:id="txt1" layoutX="100" layoutY="70"/>
<Button fx:id="btn1Save" onAction="#btn1Save" layoutX="100" layoutY="140" mnemonicParsing="false"
text="Save text"/>
<Button fx:id="btn1Load" onAction="#btn1Load" layoutX="200.0" layoutY="140" mnemonicParsing="false"
text="Send to tab 2"/>
</children>
</AnchorPane>
这些字段可能存在于控制器类中,但是
fxmloader
仍然会为给定这些fxml加载的每个fxml创建新的控制器实例。由于tab1.fxml
不包含带有fx:id=“lblController”
属性的Label
标记,因此不会将lblController
字段注入Tab1Controller
实例
最好不要扩展控制器
,而是将其引用传递给子控制器:
public class Controller {
@FXML
public Label lblController;
@FXML
private Tab1Controller tab1Controller;
@FXML
private Tab2Controller tab2Controller;
public Controller() {
System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
}
@FXML
public void initialize() {
System.out.println("CONTROLLER MAIN - INITIALIZER");
tab1Controller.initParentController(this);
tab2Controller.initParentController(this);
}
public void setSts(String sts) {
lblController.setText(sts);
}
}
请注意,对于子控制器,您实际上可以使用抽象超类实现
initParentController
方法一次。我见过这种方法,我试图分别为每个子控制器避免initialize();它可以设置主控制器。我仍然很困惑继承在这里是如何不起作用的,即使使用分离的getter/setter进行访问。@4673j加载main.fxml
,您只需获得3个不同的控制器实例:每个使用的fxml文件一个。我经常看到没有经验的fxml文件用户希望字段“神奇地”被注入到任何地方;事实并非如此。对象仅注入到与fxml文件一起使用的控制器实例中,在该实例中声明对象。我理解您的意思,我不希望使用magic,我希望了解它如何工作以及如何正确使用它,我不理解如果元素已按照注入请求初始化,我如何无法访问它?因此应用简单的继承;对于一个简单的括号,在我的程序(不是这个示例)中,我已经有9个控制器,并且还在增长@4673_j您似乎混淆了“对象/实例”和“类”。代码创建了三个不同的控制器实例。一个是控制器
的实例,一个是选项卡1控制器
的实例,一个是选项卡2控制器
的实例。使用您定义的继承,后两个当然也是Controller
的实例,因此这三个都有一个名为lblController
的字段。然而,只有当FXML文件中有一个带有fx:id=“lblController”
的元素时,该字段才被初始化,这仅在第一个实例中是正确的,而在其他两个实例中不是正确的。@James\u D这让事情更清楚了,我似乎忘记了基本内容;回到书上。谢谢你的努力。
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Button?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.control.TextField?>
<?import javafx.scene.layout.AnchorPane?>
<AnchorPane xmlns:fx="http://javafx.com/fxml/1"
xmlns="http://javafx.com/javafx/8.0.60" fx:controller="sample.ctrTab.Tab1Controller">
<children>
<Label fx:id="lbl1" layoutX="100" layoutY="50" text="Def tab 1"/>
<TextField fx:id="txt1" layoutX="100" layoutY="70"/>
<Button fx:id="btn1Save" onAction="#btn1Save" layoutX="100" layoutY="140" mnemonicParsing="false"
text="Save text"/>
<Button fx:id="btn1Load" onAction="#btn1Load" layoutX="200.0" layoutY="140" mnemonicParsing="false"
text="Send to tab 2"/>
</children>
</AnchorPane>
public class Controller {
@FXML
public Label lblController;
@FXML
private Tab1Controller tab1Controller;
@FXML
private Tab2Controller tab2Controller;
public Controller() {
System.out.println("CONTROLLER MAIN - CONSTRUCTOR");
}
@FXML
public void initialize() {
System.out.println("CONTROLLER MAIN - INITIALIZER");
tab1Controller.initParentController(this);
tab2Controller.initParentController(this);
}
public void setSts(String sts) {
lblController.setText(sts);
}
}
public class Tab1Controller {
@FXML
Label lbl1;
@FXML
TextField txt1;
@FXML
Button btn1Save, btn1Load;
public Tab1Controller() {
System.out.println("CONTROLLER TAB 1 - CONSTRUCTOR");
}
private Controller parentController;
public void initParentController(Controller controller) {
parentController = controller;
}
@FXML
void btn1Save() {
System.out.println("CONTROLLER TAB 1 - SAVE CLICK");
parentController.lblController.setText("ANYTHING");
parentController.setSts(txt1.getText());
}
@FXML
void btn1Load() {
System.out.println("CONTROLLER TAB 1 - LOAD CLICK");
}
}