我正在尝试将矩形拖动到VBox,然后再次将其从VBox内部拖动。
如果该矩形放置在目标VBox之外的任何位置,则应将其位置恢复到其所属位置:如果它是左侧VBox的子代,则将其重置到其内部;如果它是右VBox的子代,请将其重置回右VBox的内部。
第一部分工作正常,我可以将其拖动到VBox并将其添加为子级。第二部分是我遇到问题的地方。
package application;
import javafx.application.Application;
import javafx.event.Event;
import javafx.geometry.Insets;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.VBox;
import javafx.scene.shape.Rectangle;
import javafx.stage.Stage;
public class DragNodeMultipleTimes extends Application {
private Disk sourceDisk = new Disk();
private VBox targetVBox = new VBox();
public static void main(String[] args) {
Application.launch(args);
}
@Override
public void start(Stage stage) {
// Build the UI
GridPane root = getUI();
// Add the event handlers
this.addEventHandlers();
Scene scene = new Scene(root, 800, 600);
stage.setScene(scene);
stage.show();
}
private GridPane getUI() {
GridPane pane = new GridPane();
VBox sourceVBox = new VBox();
sourceDisk.setWidth(90);
sourceDisk.setHeight(20);
sourceVBox.setStyle(" -fx-border-color:red; -fx-border-width: 1; -fx-border-style: solid;");
targetVBox.setStyle(" -fx-border-color:green; -fx-border-width: 1; -fx-border-style: solid;");
sourceVBox.getChildren().add(sourceDisk);
targetVBox.getChildren().add(new Rectangle(200, 20));
pane.setHgap(200);
pane.addColumn(0, sourceVBox);
pane.addColumn(1, targetVBox);
pane.setPadding(new Insets(200, 100, 200, 100));
return pane;
}
private void addEventHandlers() {
sourceDisk.setOnMousePressed(event -> {
sourceDisk.setOrgSceneX(event.getSceneX());
sourceDisk.setOrgSceneY(event.getSceneY());
sourceDisk.setOrgTranslateX(sourceDisk.getTranslateX());
sourceDisk.setOrgTranslateY(sourceDisk.getTranslateY());
sourceDisk.setMouseTransparent(true);
event.consume();
});
sourceDisk.setOnDragDetected(event -> sourceDisk.startFullDrag());
sourceDisk.setOnMouseDragged(event -> {
double offsetX = event.getSceneX() - sourceDisk.getOrgSceneX();
double offsetY = event.getSceneY() - sourceDisk.getOrgSceneY();
double newTranslateX = sourceDisk.getOrgTranslateX() + offsetX;
double newTranslateY = sourceDisk.getOrgTranslateY() + offsetY;
sourceDisk.setTranslateX(newTranslateX);
sourceDisk.setTranslateY(newTranslateY);
});
sourceDisk.setOnMouseReleased(event -> {
//reset the position of the disk
sourceDisk.setTranslateX(0);
sourceDisk.setTranslateY(0);
sourceDisk.setMouseTransparent(false);
});
targetVBox.setOnMouseDragEntered(Event::consume);
targetVBox.setOnMouseDragOver(event -> System.out.println("VBox: mouseDragOver"));
targetVBox.setOnMouseDragReleased(event -> {
((VBox) (event.getSource())).getChildren().add(0, sourceDisk);
//reset translate values
sourceDisk.setTranslateX(0);
sourceDisk.setTranslateY(0);
sourceDisk.setMouseTransparent(false);
});
}
private class Disk extends Rectangle {
private double orgSceneX;
private double orgSceneY;
private double orgTranslateX;
private double orgTranslateY;
public double getOrgSceneX() {
return orgSceneX;
}
public void setOrgSceneX(double orgSceneX) {
this.orgSceneX = orgSceneX;
}
public double getOrgSceneY() {
return orgSceneY;
}
public void setOrgSceneY(double orgSceneY) {
this.orgSceneY = orgSceneY;
}
public double getOrgTranslateX() {
return orgTranslateX;
}
public void setOrgTranslateX(double orgTranslateX) {
this.orgTranslateX = orgTranslateX;
}
public double getOrgTranslateY() {
return orgTranslateY;
}
public void setOrgTranslateY(double orgTranslateY) {
this.orgTranslateY = orgTranslateY;
}
}
}
重现此问题的步骤:
1)将左矩形拖到右矩形的顶部(在VBox内)
2)将新添加的矩形拖动到VBox外部的位置
3)释放鼠标按钮
问题出现在第2步。我期望VBox不再从鼠标获取任何事件,因为它离开了边界。从控制台输出中可以看出,这不会发生(仍在调用onMouseDragOver)。
在第3步,引发了一个异常,但这是因为它试图将相同的矩形添加回VBox。
我期望的是,如果我将矩形拖动到右侧VBox的外部并释放鼠标,则该矩形会将其位置重置为回到右侧VBox的内部。
基本上,我要弄清楚的是,即使光标不在VBox的范围内,它仍然会收到MOUSE_DRAGGED事件呢?
我也乐于接受其他实现此结果的方法的建议。
最佳答案
我不太确定,但我有一个假设。
磁盘作为VBox的子级添加,因此看起来像是磁盘的一部分,并且相同事件适用于VBox应用于磁盘。因此,由于磁盘是VBox的一部分,所以当您拖动它的子项时,鼠标仍然在VBox中
我认为是这样的:如果孩子收到事件,所有父母也都会收到。
因此,只要将磁盘拖到VBox中,就需要将其从VBox中删除。
关于java - 使用JavaFX多次拖动节点,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/56416801/