SceneBuilder 不会加载我的自定义控件,它通过 FXML 引用另一个自定义控件
SceneBuilder won't load my custom control which references another custom control via FXML
我创建了一个基于 FXML 的自定义控件,它又引用了另一个基于 FXML 的自定义控件。当我在 eclipse 中加载它们时它们都工作得很好,但是当我尝试将它们导入 SceneBuilder 时,外部控件(包含另一个的控件)没有正确导入。
这是一个大大简化的例子:
Widget.java:
package example;
import java.util.logging.Level;
import java.util.logging.Logger;
import javafx.fxml.FXMLLoader;
import javafx.scene.layout.Pane;
public class Widget extends Pane{
public Widget() {
Logger logger = Logger.getLogger(Widget.class.getName());
try {
FXMLLoader loader= new FXMLLoader(Widget.class.getResource("Widget.fxml"));
Pane pane = loader.load();
this.getChildren().add(pane);
}
catch(Exception e) {
logger.log(Level.SEVERE, "Failed to load Widget", e);
}
}
}
Widget.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.control.Label?>
<?import example.SubWidget?>
<FlowPane xmlns:fx="http://javafx.com/fxml/1" >
<children>
<Label text="Widget"/>
<SubWidget></SubWidget>
</children>
</FlowPane>
子Widget.java:
package example;
import java.util.logging.Level;
import java.util.logging.Logger;
import javafx.fxml.FXMLLoader;
import javafx.scene.layout.Pane;
public class SubWidget extends Pane{
public SubWidget() {
Logger logger = Logger.getLogger(SubWidget.class.getName());
try {
FXMLLoader loader= new FXMLLoader(SubWidget.class.getResource("SubWidget.fxml"));
Pane pane = loader.load();
this.getChildren().add(pane);
}
catch(Exception e) {
logger.log(Level.SEVERE, "Failed to load SubWidget", e);
}
}
}
子Widget.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.control.Label?>
<FlowPane xmlns:fx="http://javafx.com/fxml/1" >
<children>
<Label text="Subwidget"/>
</children>
</FlowPane>
我将它们导出到 jar 并尝试通过 SceneBuilder 的 jar 导入功能导入它们。当我这样做时,SubWidget
被导入并且看起来很好。 Widget
也将显示为控件,但它将完全为空。
当我检查日志文件时,我发现这是由于 FXML 加载程序无法找到 SubWidget class 文件:
Oct 29, 2020 4:28:30 PM example.Widget <init>
SEVERE: Failed to load Widget
javafx.fxml.LoadException:
file:/C:/Users/nate/AppData/Roaming/Scene%20Builder/Library/TestFxControl.jar!/example/Widget.fxml
at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2601)
at javafx.fxml.FXMLLoader.importClass(FXMLLoader.java:2848)
at javafx.fxml.FXMLLoader.processImport(FXMLLoader.java:2692)
at javafx.fxml.FXMLLoader.processProcessingInstruction(FXMLLoader.java:2661)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2517)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2441)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2409)
at example.Widget.<init>(Widget.java:27)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Source)
at java.lang.reflect.Constructor.newInstance(Unknown Source)
at java.lang.Class.newInstance(Unknown Source)
at sun.reflect.misc.ReflectUtil.newInstance(Unknown Source)
at javafx.fxml.FXMLLoader$InstanceDeclarationElement.constructValue(FXMLLoader.java:1009)
at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:746)
at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2707)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2527)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2425)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.instantiateWithFXMLLoader(JarExplorer.java:110)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.exploreEntry(JarExplorer.java:160)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.explore(JarExplorer.java:70)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.exploreAndUpdateLibrary(LibraryFolderWatcher.java:325)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.runDiscovery(LibraryFolderWatcher.java:138)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.run(LibraryFolderWatcher.java:92)
at java.lang.Thread.run(Unknown Source)
Caused by: java.lang.ClassNotFoundException: example.SubWidget
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at javafx.fxml.FXMLLoader.loadTypeForPackage(FXMLLoader.java:2916)
at javafx.fxml.FXMLLoader.loadType(FXMLLoader.java:2905)
at javafx.fxml.FXMLLoader.importClass(FXMLLoader.java:2846)
... 24 more
在我看来,FXML 加载器使用的 class 加载器不包含其他自定义组件,这很奇怪,因为如果我在 Widget
[=49] 中绕过 FXML =] 并做到了:
public class Widget extends FlowPane{
public Widget() {
this.getChildren().addAll(new Label("Widget"), new SubWidget());
}
}
然后它在 SceneBuilder 中加载得很好。所以看起来实际加载 Widget
的 ClassLoader
和 FXMLLoader
使用的不一样。
这可能与 this unanswered question 有关,尽管它们并不完全相同。
我遇到了同样的问题,并创建了一个解决方案,它虽然不漂亮但有效。
为无法加载到 Scene Builder 的控件创建一个包含空白 类 的 jar,例如:
package com.junglefinance.gui.controls;
import javafx.fxml.FXML;
import javafx.scene.control.TextField;
public class CategoryTextField extends TextField {
@FXML private TextField textField;
public CategoryTextField() {
}
}
确保包与目标控件的包匹配。
将此 jar 加载到 Scene Builder 中。这将允许 Scene Builder 生成准备加载到程序中的 FXML。当它被加载时,它将选择你的目标控件而不是导入的虚拟控件(在这种情况下)
<?import com.junglefinance.gui.controls.CategoryTextField?>
注意包裹。
我创建了一个基于 FXML 的自定义控件,它又引用了另一个基于 FXML 的自定义控件。当我在 eclipse 中加载它们时它们都工作得很好,但是当我尝试将它们导入 SceneBuilder 时,外部控件(包含另一个的控件)没有正确导入。
这是一个大大简化的例子:
Widget.java:
package example;
import java.util.logging.Level;
import java.util.logging.Logger;
import javafx.fxml.FXMLLoader;
import javafx.scene.layout.Pane;
public class Widget extends Pane{
public Widget() {
Logger logger = Logger.getLogger(Widget.class.getName());
try {
FXMLLoader loader= new FXMLLoader(Widget.class.getResource("Widget.fxml"));
Pane pane = loader.load();
this.getChildren().add(pane);
}
catch(Exception e) {
logger.log(Level.SEVERE, "Failed to load Widget", e);
}
}
}
Widget.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.control.Label?>
<?import example.SubWidget?>
<FlowPane xmlns:fx="http://javafx.com/fxml/1" >
<children>
<Label text="Widget"/>
<SubWidget></SubWidget>
</children>
</FlowPane>
子Widget.java:
package example;
import java.util.logging.Level;
import java.util.logging.Logger;
import javafx.fxml.FXMLLoader;
import javafx.scene.layout.Pane;
public class SubWidget extends Pane{
public SubWidget() {
Logger logger = Logger.getLogger(SubWidget.class.getName());
try {
FXMLLoader loader= new FXMLLoader(SubWidget.class.getResource("SubWidget.fxml"));
Pane pane = loader.load();
this.getChildren().add(pane);
}
catch(Exception e) {
logger.log(Level.SEVERE, "Failed to load SubWidget", e);
}
}
}
子Widget.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import javafx.scene.layout.FlowPane?>
<?import javafx.scene.control.Label?>
<FlowPane xmlns:fx="http://javafx.com/fxml/1" >
<children>
<Label text="Subwidget"/>
</children>
</FlowPane>
我将它们导出到 jar 并尝试通过 SceneBuilder 的 jar 导入功能导入它们。当我这样做时,SubWidget
被导入并且看起来很好。 Widget
也将显示为控件,但它将完全为空。
当我检查日志文件时,我发现这是由于 FXML 加载程序无法找到 SubWidget class 文件:
Oct 29, 2020 4:28:30 PM example.Widget <init>
SEVERE: Failed to load Widget
javafx.fxml.LoadException:
file:/C:/Users/nate/AppData/Roaming/Scene%20Builder/Library/TestFxControl.jar!/example/Widget.fxml
at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2601)
at javafx.fxml.FXMLLoader.importClass(FXMLLoader.java:2848)
at javafx.fxml.FXMLLoader.processImport(FXMLLoader.java:2692)
at javafx.fxml.FXMLLoader.processProcessingInstruction(FXMLLoader.java:2661)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2517)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2441)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2409)
at example.Widget.<init>(Widget.java:27)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(Unknown Source)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(Unknown Source)
at java.lang.reflect.Constructor.newInstance(Unknown Source)
at java.lang.Class.newInstance(Unknown Source)
at sun.reflect.misc.ReflectUtil.newInstance(Unknown Source)
at javafx.fxml.FXMLLoader$InstanceDeclarationElement.constructValue(FXMLLoader.java:1009)
at javafx.fxml.FXMLLoader$ValueElement.processStartElement(FXMLLoader.java:746)
at javafx.fxml.FXMLLoader.processStartElement(FXMLLoader.java:2707)
at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2527)
at javafx.fxml.FXMLLoader.load(FXMLLoader.java:2425)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.instantiateWithFXMLLoader(JarExplorer.java:110)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.exploreEntry(JarExplorer.java:160)
at com.oracle.javafx.scenebuilder.kit.library.util.JarExplorer.explore(JarExplorer.java:70)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.exploreAndUpdateLibrary(LibraryFolderWatcher.java:325)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.runDiscovery(LibraryFolderWatcher.java:138)
at com.oracle.javafx.scenebuilder.kit.library.user.LibraryFolderWatcher.run(LibraryFolderWatcher.java:92)
at java.lang.Thread.run(Unknown Source)
Caused by: java.lang.ClassNotFoundException: example.SubWidget
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at javafx.fxml.FXMLLoader.loadTypeForPackage(FXMLLoader.java:2916)
at javafx.fxml.FXMLLoader.loadType(FXMLLoader.java:2905)
at javafx.fxml.FXMLLoader.importClass(FXMLLoader.java:2846)
... 24 more
在我看来,FXML 加载器使用的 class 加载器不包含其他自定义组件,这很奇怪,因为如果我在 Widget
[=49] 中绕过 FXML =] 并做到了:
public class Widget extends FlowPane{
public Widget() {
this.getChildren().addAll(new Label("Widget"), new SubWidget());
}
}
然后它在 SceneBuilder 中加载得很好。所以看起来实际加载 Widget
的 ClassLoader
和 FXMLLoader
使用的不一样。
这可能与 this unanswered question 有关,尽管它们并不完全相同。
我遇到了同样的问题,并创建了一个解决方案,它虽然不漂亮但有效。
为无法加载到 Scene Builder 的控件创建一个包含空白 类 的 jar,例如:
package com.junglefinance.gui.controls;
import javafx.fxml.FXML;
import javafx.scene.control.TextField;
public class CategoryTextField extends TextField {
@FXML private TextField textField;
public CategoryTextField() {
}
}
确保包与目标控件的包匹配。
将此 jar 加载到 Scene Builder 中。这将允许 Scene Builder 生成准备加载到程序中的 FXML。当它被加载时,它将选择你的目标控件而不是导入的虚拟控件(在这种情况下)
<?import com.junglefinance.gui.controls.CategoryTextField?>
注意包裹。