TOFRONT()はサポートされていないPerationExceptionを投げます
質問
StackPane
を作成したい場合は、追加されているものは何でも、自分の選択のノードは常に正面にあります(注文に注意してください。何でも追加されます。)
これを行うには、該当するtoFront()
オブジェクトの子リストにリスナーを配置するだけで、該当するノード上のStackPane
を呼び出します。
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 7では、これはちょうどうまく機能します。ただし、JFX8(最新のビルド)では、次のように失敗します。
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)
.
とはい、toFront()
は実際にtest.Test$1.onChanged(Test.java:34)
を参照しています。
これはバグと見なされるべきであるか、私はこのようにして物事を達成しようとしていることに気付かないルールを破っていますか? r1.toFront();
メソッドが実行されていた間にリストが変更されていたのか疑問には、onChanged()
もリストの内容を変更します。したがって、javadocは、toFront()
へのJavadocは明らかに言っています。
の後、 ObservableListに変更されました。
(太字は私のものです。)
編集:この時点で、それがバグであることをもっと確実にしているので、関連するバグ報告はこちら
解決
現在同じリストの別の変更イベントを取り扱っているイベントハンドラ内の(JavaFx)リストを変更することを許可されていないようです。これは合理的なようですが自明ではないので、その場合にはより明白な例外があるはずです。
残念ながら、散逸以外の例外はJavaFXで非常に一般的です。
幸運な解決策/回避策はかなり簡単です。
root.getChildren().addListener(new ListChangeListener<Node>() {
@Override
public void onChanged(ListChangeListener.Change<? extends Node> change) {
Platform.runLater(new Runnable() {
public void run() { r1.toFront(); }
});
}
});
.
SideNote:コンポーネントがすでに正面にある場合は、r1.toFront
は何もしません。これは無限ループを防ぎます。それにもかかわらず、これは文書に明示的に言及されていないので、あなたはそれに頼らないかもしれません。