我用JavaFX创建了一个简单的无边界应用程序。现在,我创建了一个Label,它应该替换缺少的“关闭”按钮。实际上,我的尝试没有用,因为我无法与FXMLController进行交互。
我已经尝试将FXMLController添加到我的* .fxml文件中。我试图将控制器插入到VBox和AnchorPane中,但两者都无法正常工作,我真的不明白。
LabelCloseTest.fxml
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.control.Label?>
<?import javafx.scene.layout.AnchorPane?>
<?import javafx.scene.layout.VBox?>
<?import javafx.scene.text.Font?>
<VBox prefHeight="720.0" prefWidth="1025.0"
xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1">
<children>
<AnchorPane maxHeight="-1.0" maxWidth="-1.0" prefHeight="-1.0" prefWidth="-1.0" style="-fx-background-color: #303136;" VBox.vgrow="ALWAYS" fx:controller="com.mycompany.testing.FXMLDocumentController">
<children>
<Label fx:id="labelTest" layoutX="1007.0" layoutY="5.0" prefHeight="10.0" prefWidth="5.0" text="X" textFill="WHITE">
<font>
<Font name="System Bold" size="13.0" />
</font>
</Label>
</children>
</AnchorPane>
</children>
</VBox>
FXMLDocumentController.java
package com.mycompany.testing;
import javafx.event.EventHandler;
import javafx.fxml.FXML;
import javafx.scene.control.Label;
import javafx.scene.input.MouseEvent;
public class FXMLDocumentController {
@FXML
private Label labelTest;
@FXML
public void closeLabelPressed() {
labelTest.setOnMouseClicked(new EventHandler<MouseEvent>() {
public void handle(MouseEvent event) {
System.exit(0);
}
});
}
}
LabelTest.java
package com.mycompany.testing;
import javafx.application.Application;
import javafx.fxml.FXMLLoader;
import javafx.scene.Parent;
import javafx.scene.Scene;
import javafx.stage.Stage;
import javafx.stage.StageStyle;
public class LabelTest extends Application {
public static void main (String[] args) {
launch(args);
}
@Override
public void start(Stage primaryStage) throws Exception {
primaryStage.initStyle(StageStyle.UNDECORATED);
FXMLLoader loader = new FXMLLoader(getClass().getResource("/fxml/LabelCloseTest.fxml"));
Parent root = loader.load();
FXMLDocumentController fdc = new FXMLDocumentController();
fdc.closeLabelPressed();
Scene scene = new Scene(root);
primaryStage.setScene(scene);
primaryStage.show();
}
}
Stacktrace
Exception in Application start method
Exception in thread "main" java.lang.RuntimeException: Exception in Application start method
at com.sun.javafx.application.LauncherImpl.launchApplication1(LauncherImpl.java:900)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication$2(LauncherImpl.java:195)
at java.base/java.lang.Thread.run(Thread.java:834)
Caused by: javafx.fxml.LoadException: fx:controller can only be applied to root element.
/D:/Workspace/SocketClient/target/classes/fxml/LabelCloseTest.fxml:10
at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2621)
at javafx.fxml.FXMLLoader$ValueElement.processAttribute(FXMLLoader.java:917)
at javafx.fxml.FXMLLoader$InstanceDeclarationElement.processAttribute(FXMLLoader.java:980)
at javafx.fxml.FXMLLoader$Element.processStartElement(FXMLLoader.java:227)
at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:752)
at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2722)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2552)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2466)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2435)
at com.mycompany.testing.LabelTest.start(LabelTest.java:25)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$9(LauncherImpl.java:846)
at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
at java.base/java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
... 1 more
我希望应用程序启动,当我按下“ X”标签时,应用程序将关闭。
快速提醒:我是JavaFX的新手,所以我还要感谢每一个JavaFX教程的建议。
答案 0 :(得分:3)
我试图将控制器插入到VBox和AnchorPane中,但是两者都无法正常工作,我真的不明白。
您提到尝试将fx:controller
属性同时放在VBox
元素和AnchorPane
元素上。您显示的代码是您尝试将其添加到AnchorPane
元素中的尝试。照原样在计算机上运行代码会导致您观察到LoadException
。通过将fx:controller
属性放在VBox
元素(root element)中可以解决此问题:
<VBox prefHeight="720.0" prefWidth="1025.0"
xmlns="http://javafx.com/javafx/10.0.1" xmlns:fx="http://javafx.com/fxml/1"
fx:controller="com.mycompany.testing.FXMLDocumentController">
<!--- Rest of FXML file omitted for brevity -->
注意:我最初注意到另一个问题,即您没有关闭<VBox>
或<children>
元素。但是,这些结束标记始终存在。他们只是没有出现在您的问题中,as pointed out by fabian。我忽略了检查。
但是,这会导致不同例外。
// Beginning of stack trace omitted for brevity...
Caused by: java.lang.NullPointerException
at com.mycompany.testing.FXMLDocumentController.closeLabelPressed(FXMLDocumentController.java:16)
at com.mycompany.testing.LabelTest.start(LabelTest.java:28)
at javafx.graphics/com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$9(LauncherImpl.java:846)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
at java.base/java.security.AccessController.doPrivileged(Native Method)
at javafx.graphics/com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
at javafx.graphics/com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
at javafx.graphics/com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at javafx.graphics/com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
... 1 more
此问题最终是由LabelTest#start
中的以下代码引起的:
FXMLLoader loader = new FXMLLoader(getClass().getResource("/fxml/LabelCloseTest.fxml"));
Parent root = loader.load();
FXMLDocumentController fdc = new FXMLDocumentController(); // the real problem
fdc.closeLabelPressed(); // accesses the labelTest field which is null
由于您使用的是fx:controller
,因此FXMLLoader
将创建它的 own 实例FXMLDocumentController
。在这种情况下,注入了@FXML
带注释的字段。另一方面,您将创建您自己的实例,该实例将没有注入字段。您应该使用FXMLDocumentController
创建的FXMLLoader
实例。
FXMLLoader loader = new FXMLLoader(getClass().getResource("/fxml/LabelCloseTest.fxml"));
Parent root = loader.load();
// gets instance from FXMLLoader, must be called after load()
FXMLDocumentController fdc = loader.getController();
fdc.closeLabelPressed();
但是,尽管您的代码(到目前为止已修复)可以实现您想要的功能,但这并不是正确的方法。方法closeLabelPressed
看起来像您打算将其用作事件处理程序,因为您已用@FXML
对其进行了注释。但是,您没有将此方法与FXML文件链接,而是该方法将EventHandler
添加到Label
。正确设置后,您根本不应从closeLabelPressed
呼叫LabelTest
。
您应该做的第一件事是通过FXML文件将方法链接到Label
。
<!-- previous omitted for brevity -->
<Label fx:id="labelTest" layoutX="1007.0" layoutY="5.0" prefHeight="10.0" prefWidth="5.0" text="X"
textFill="WHITE" onMousePressed="#closeLabelPressed">
<!-- rest omitted for brevity -->
注意onMousePressed
属性。
第二件事是更改closeLabelPressed
中的FXMLDocumentController
方法。
package com.mycompany.testing;
import javafx.application.Platform;
import javafx.fxml.FXML;
import javafx.scene.control.Label;
import javafx.scene.input.MouseEvent;
public class FXMLDocumentController {
@FXML
private Label labelTest;
@FXML
private void closeLabelPressed(MouseEvent event) {
Platform.exit();
}
}
现在按下Label
时将调用该方法。在内部,FXMLLoader
将onMousePressed
的{{1}}属性设置为调用labelTest
的{{1}}。我也将EventHandler
的使用更改为closeLabelPressed
。后者将允许JavaFX运行时正常关闭(允许您在System.exit(0)
之类的地方进行清理),而前者将立即终止JVM(并且不会正常返回)。 注意:不再需要注入Platform.exit()
。
第三件事是在Application#stop
方法中删除对Label
的调用。
这些更改之后,您的代码应该可以正常工作。
关于询问教程(有关Stack Overflow的话题,请参阅help center),您可能可以通过搜索Google找到许多教程。有关“官方”资源,请参见JavaFX: Getting started with JavaFX(最近更新为JavaFX 8)和Introduction to FXML。堆栈溢出中也有many questions与FXML相关的内容。