使用JavaFX多次拖动节点

时间:2019-06-02 16:20:49

标签: java javafx drag-and-drop

我正在尝试将矩形拖动到VBox,然后再次将其从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事件?
我也乐于接受其他实现此结果的方法的建议。

1 个答案:

答案 0 :(得分:0)

我不太确定,但我有一个假设。 磁盘作为VBox的子级添加,因此看起来像磁盘的一部分,并且相同事件适用于VBox应用于磁盘。因此,由于磁盘是VBox的一部分,所以当您拖动它的孩子时,鼠标仍然在VBox中

我想是这样的:如果一个孩子收到一个事件,那么所有父母也都会收到它。

因此,只要将磁盘拖到VBox之外,就需要将其从VBox中删除。