我想创建一个StackPane
,无论添加什么,我选择的节点总是在前面(而不是要小心我添加内容的顺序,或者记得调用{{无论何时添加任何内容,都会在该特定节点上添加。)
为了做到这一点,我只是将一个监听器放在相关toFront()
对象的子列表中,这样无论何时发生任何变化,它都会在相关节点上调用StackPane
,例如:< / p>
toFront()
在Java 7中,这很好用。但是,在JFX8(刚刚下载的最新版本)中,它失败并显示以下内容:
public class Test extends Application {
@Override
public void start(Stage stage) {
StackPane root = new StackPane();
final Rectangle r1 = new Rectangle(50, 50);
root.getChildren().add(r1);
root.getChildren().addListener(new ListChangeListener<Node>() {
@Override
public void onChanged(ListChangeListener.Change<? extends Node> change) {
try {
while(change.next()) {
if(change.wasAdded()) {
r1.toFront();
}
}
}
catch(Exception ex) {
ex.printStackTrace();
}
}
});
root.getChildren().add(new Rectangle(50, 50));
stage.setScene(new Scene(root));
stage.show();
}
}
是的,java.lang.UnsupportedOperationException
at java.util.Collections$UnmodifiableList.add(Collections.java:1374)
at javafx.collections.ListChangeBuilder.nextRemove(ListChangeBuilder.java:208)
at javafx.collections.ObservableListBase.nextRemove(ObservableListBase.java:150)
at javafx.collections.ModifiableObservableListBase.remove(ModifiableObservableListBase.java:181)
at com.sun.javafx.collections.VetoableListDecorator.remove(VetoableListDecorator.java:284)
at com.sun.javafx.collections.VetoableListDecorator.remove(VetoableListDecorator.java:209)
at javafx.scene.Parent.impl_toFront(Parent.java:624)
at javafx.scene.Node.toFront(Node.java:1713)
at test.Test$1.onChanged(Test.java:34)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:315)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:72)
at com.sun.javafx.collections.VetoableListDecorator$1.onChanged(VetoableListDecorator.java:77)
at com.sun.javafx.collections.ListListenerHelper$Generic.fireValueChangedEvent(ListListenerHelper.java:315)
at com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent(ListListenerHelper.java:72)
at javafx.collections.ObservableListBase.fireChange(ObservableListBase.java:233)
at javafx.collections.ListChangeBuilder.commit(ListChangeBuilder.java:482)
at javafx.collections.ListChangeBuilder.endChange(ListChangeBuilder.java:541)
at javafx.collections.ObservableListBase.endChange(ObservableListBase.java:205)
at javafx.collections.ModifiableObservableListBase.add(ModifiableObservableListBase.java:155)
at java.util.AbstractList.add(AbstractList.java:108)
at com.sun.javafx.collections.VetoableListDecorator.add(VetoableListDecorator.java:200)
at test.Test.start(Test.java:41)
at com.sun.javafx.application.LauncherImpl$8.run(LauncherImpl.java:837)
at com.sun.javafx.application.PlatformImpl$7.run(PlatformImpl.java:331)
at com.sun.javafx.application.PlatformImpl$6$1.run(PlatformImpl.java:297)
at com.sun.javafx.application.PlatformImpl$6$1.run(PlatformImpl.java:294)
at java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl$6.run(PlatformImpl.java:294)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
at com.sun.glass.ui.win.WinApplication.access$300(WinApplication.java:39)
at com.sun.glass.ui.win.WinApplication$4$1.run(WinApplication.java:112)
at java.lang.Thread.run(Thread.java:744)
确实是指test.Test$1.onChanged(Test.java:34)
。
这是否被视为一个错误,或者我是否打破了一些我不知道的规则,试图以这种方式实现目标?我确实想知道在r1.toFront();
方法执行时列表是否仍然被更改,onChanged()
也会更改列表内容,因此异常 - 但是toFront()
的Javadoc清楚地说:
在之后调用对ObservableList进行了更改。
(Bolding是我的。)
编辑:此时我更确定这是一个错误,所以相关的错误报告是here。
答案 0 :(得分:3)
似乎不允许修改当前正在处理同一列表的另一个(上一个)修改事件的事件处理程序中的(JavaFX)列表。虽然这似乎是合理的,但这并不是不证自明的,因此在这种情况下应该有一个更明显的例外。
不幸的是,非Speeking异常在JavaFX中很常见。
幸运的是,解决方案/解决方法非常简单:通过r1.toFront
调用修改代码(此处:Platform.runLater()
),它会延迟您的修改发生在原始事件之后:
root.getChildren().addListener(new ListChangeListener<Node>() {
@Override
public void onChanged(ListChangeListener.Change<? extends Node> change) {
Platform.runLater(new Runnable() {
public void run() { r1.toFront(); }
});
}
});
旁注:如果组件已经在前面,toFront
什么都不做。这可以防止无限循环。然而,由于文档中没有明确提及,您可能不会依赖它。