限制3D对象鼠标将移动拖动到JavaFX中的平面

时间:2015-03-13 13:55:12

标签: 3d javafx axis drag javafx-3d

我使用JavaFX通过鼠标拖动来移动3D立方体。立方体应保持在x和z轴跨越的平面上。我的解决方案运行得相当不错,但是如果用鼠标移动立方体太快或遇到具有一定深度的对象(y轴),则假设鼠标在y轴和立方体上移动开始向前或向后跳跃。有没有办法将鼠标限制在xz平面?一个更复杂的解决方案是将y长度投射回xz平面,但我不知道如何。 我看了JavaFX Moving 3D Objects,但无法适应我的情况。

到目前为止我的代码:

    private volatile double relMousePosX;
    private volatile double relMousePosZ;

    public void enableDragMove(PaneBox paneBox) {
        Group paneBoxGroup = paneBox.get();

        paneBoxGroup.setOnMousePressed((MouseEvent me) -> {
            if(isSelected(paneBox) && MouseButton.PRIMARY.equals(me.getButton())) {
                relMousePosX = me.getX();
                relMousePosZ = me.getZ();
            }
        });

        paneBoxGroup.setOnMouseDragged((MouseEvent me) -> {
            if(paneBoxGroup.focusedProperty().get() && MouseButton.PRIMARY.equals(me.getButton())) {
                setDragInProgress(paneBox, true);
                System.out.println(me.getY()); // should stay small value, but jumps to higher values at times, creating the problem.
                paneBoxGroup.setCursor(Cursor.MOVE);
                paneBox.setTranslateX(paneBox.getTranslateX() + (me.getX() - relMousePosX));
                paneBox.setTranslateZ(paneBox.getTranslateZ() + (me.getZ() - relMousePosZ));
            }
        });

        paneBoxGroup.setOnMouseReleased((MouseEvent me) -> {
            if(paneBoxGroup.focusedProperty().get() && MouseButton.PRIMARY.equals(me.getButton())) {
                setDragInProgress(paneBox, false);
                paneBoxGroup.setCursor(Cursor.DEFAULT);
            }
        });
   }

1 个答案:

答案 0 :(得分:4)

正如您所提到的,3D拖动有两种可能的方法:

  • 纯粹的阻力&像你提议的那样下降
  • 取消投影指示,例如此question

对于拖放事件,我们可以在事件开始和结束时检测3D形状上的拖动事件。

这里的诀窍是,不要在形状上的拖动事件之间聆听,而是在可能的目标上拖拽事件。

通过定义目标,您可以轻松限制形状移动,因为只有在仅在此目标上拖动时才会更新其位置。

因此,如果您想限制飞机上的移动,可以使用Rectangle作为目标。

此代码段显示了此方法的工作原理:

@Override 
public void start(Stage stage) {
    final PerspectiveCamera cam = new PerspectiveCamera();
    cam.setFieldOfView(20);
    cam.setFarClip(10000);
    cam.setNearClip(0.01);
    cam.getTransforms().addAll(new Rotate(60,Rotate.X_AXIS),new Translate(-200,-200,300));

    final Group root = new Group();

    final Box floor = new Box(500, 500, 1);
    floor.setTranslateX(200);
    floor.setTranslateY(200);
    floor.setTranslateZ(50);
    floor.setMaterial(new PhongMaterial(Color.YELLOW));
    root.getChildren().add(floor);

    final Box box = new Box(50, 50, 50);
    box.setMaterial(new PhongMaterial(Color.RED));
    root.getChildren().add(box);

    final Rectangle rectangle = new Rectangle(400, 400, Color.TRANSPARENT);
    rectangle.setMouseTransparent(true);
    rectangle.setDepthTest(DepthTest.DISABLE);
    root.getChildren().add(rectangle);

    // D&D starts
    box.setOnDragDetected((MouseEvent event)-> {
        box.setMouseTransparent(true);
        rectangle.setMouseTransparent(false);
        box.setCursor(Cursor.MOVE);
        box.startFullDrag();
    });

    // D&D ends
    box.setOnMouseReleased((MouseEvent event)-> {
        box.setMouseTransparent(false);
        rectangle.setMouseTransparent(true);
        box.setCursor(Cursor.DEFAULT);
    });

    // While D&D, only confined to the rectangle
    rectangle.setOnMouseDragOver((MouseDragEvent event)-> {
        Point3D coords = event.getPickResult().getIntersectedPoint();
        coords = rectangle.localToParent(coords);
        box.setTranslateX(coords.getX());
        box.setTranslateY(coords.getY());
        box.setTranslateZ(coords.getZ());
    });

    final Scene scene = new Scene(root, 800, 600, true);
    scene.setCamera(cam);

    stage.setScene(scene);
    stage.setTitle("JavaFX 3D Drag&Drop");
    stage.show();
}

如果您运行它,您将看到可以选择该框并将其拖到地板上。

在这张图片中,我在矩形中添加了一些颜色和笔触,以查看拖动的限制。

3D drag

另请注意鼠标中的更改在框和矩形中都是透明的。只有在拖动过程中,矩形才是鼠标透明的。