无法使用@FXML anotation访问超类Controller中的元素

时间:2016-12-12 17:26:08

标签: java javafx fxml

我正在尝试使用JavaFX为我的程序设置一个简单的示例。 我想要的是一个带有main.fxml的Controller;那么main.fxml将有一个TabbedPane作为root,有2个标签(tab1.fxml和tab2.fxml),每个标签都有它的控制器(Tab1Controller和Tab2Controller)。

现在这可能是问题,但我不明白为什么会出现问题:

Tab1Controller和Tab2Controller都扩展了Controller ;因为它们共享各种被操纵的字段,例如需要不断更新的底部状态栏。

到目前为止,我设法使用所有控制器和.fxml。

当我尝试从其中一个子类设置Controller中的标签文本时,它只指向null,但标签在gui中以其默认文本初始化。 正确编译并且lblController似乎与main.fxml中的fx:id正确链接。

非常感谢任何帮助/链接。

我一直在看各种帖子,但是唯一一个接近我需要的帖子: JavaFX 8, controllers inheritance

主:

package sample;

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

public class Main extends Application {

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

    @Override
    public void start(Stage primaryStage) throws Exception {
        Parent root = FXMLLoader.load(getClass().getResource("../view/main.fxml"));
        primaryStage.setTitle("Hello World");
        primaryStage.setScene(new Scene(root, 600, 700));
        primaryStage.show();
    }
}

主控制器:

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>

Tab1Controller

package sample.ctrTab;

import javafx.fxml.FXML;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import sample.Controller;


public class Tab1Controller extends Controller {
    @FXML
    Label lbl1;
    @FXML
    TextField txt1;
    @FXML
    Button btn1Save, btn1Load;

    public Tab1Controller() {
        super();
        System.out.println("CONTROLLER TAB 1 - CONSTRUCTOR");
    }

    @FXML
    void btn1Save() {
        System.out.println("CONTROLLER TAB 1 - SAVE CLICK");
//        lblController.setText("ANYTHING"); //NULL POINTER HERE
//        setSts(txt1.getText()); //OR HERE
    }

    @FXML
    void btn1Load() {
        System.out.println("CONTROLLER TAB 1 - LOAD CLICK");
    }

    protected void setSts(String sts) {
        super.setSts(sts);
    }
}

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>

1 个答案:

答案 0 :(得分:2)

这些字段可能存在于控制器类中,但是FXMLLoader仍会为给定这些fxmls的每个fxml创建新的控制器实例。由于tab1.fxml不包含Label标记fx:id="lblController"作为属性,因此lblController字段永远不会注入Tab1Controller实例。

最好不要扩展Controller,而是将它的引用传递给子控制器:

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");
    }

}

请注意,对于子控制器,您实际上可以使用抽象超类只实现initParentController方法一次。